>  기사  >  백엔드 개발  >  PHP가 공유 메모리를 생성하여 로드를 줄이는 방법에 대한 자세한 예

PHP가 공유 메모리를 생성하여 로드를 줄이는 방법에 대한 자세한 예

藏色散人
藏色散人앞으로
2022-11-03 17:03:033953검색

이 글에서는 PHP에서 공유 메모리를 생성하여 부하를 줄이는 방법을 소개합니다. 여기서는 shmop 함수를 이해해야 합니다. 글을 따라가면서 알아보세요~

PHP에는 메모리 공유를 위한 두 가지 인터페이스가 있습니다. 하나는 실제로 변수 공유이며 객체 변수를 저장하기 전에 직렬화하는 shm입니다. 사용하기는 매우 편리하지만 효율성을 우선시하는 메모리 액세스 작업에는 직렬화된 저장소가 의미가 없습니다. 다른 하나는 shmop인데 Linux와 Windows에 공통적으로 존재하지만 shm보다 기능이 약하다. Linux에서는 shm* 계열의 함수를 호출하여 직접 구현하는 반면, Winodows에서는 시스템을 캡슐화하여 구현하기도 한다. 함수도 같은 호출을 했습니다.

공유 메모리 세그먼트를 생성하려면 shmop 함수를 사용해야 하므로 전제 조건은 확장 기능을 활성화하는 것입니다. [추천 학습: PHP 동영상 튜토리얼]

shmop 주요 기능

shmop_open(공유 메모리 블록 생성 또는 열기), shmop_write(공유 메모리 블록에 데이터 쓰기), shmop_read(공유 메모리 블록에서 읽기) data), shmop_size(공유 메모리 블록의 크기 가져오기), shmop_close(공유 메모리 블록 닫기), shmop_delete(공유 메모리 블록 삭제)

<?php
//创建一块共享内存
$shm_key = 0x4337b101;
$shm_id = @shmop_open($shm_key, &#39;c&#39;, 0644, 1024);
//读取并写入数据
$data = shmop_read($shm_id, 0, 1024);
shmop_write($shm_id, json_encode($data), 0);
$size = shmop_size($shm_id); //获取内存中实际数据占用大小
//关闭内存块,并不会删除共享内存,只是清除 PHP 的资源
shmop_close($shm_id);

shmop_open(메모리 세그먼트 생성)

가장 먼저 할 일은 이 함수에 나타나는 것은 시스템 ID 매개변수입니다. 이는 시스템의 공유 메모리 세그먼트를 식별하는 번호입니다. 두 번째 매개변수는 액세스 모드로, fopen 함수의 액세스 모드와 매우 유사합니다. 4가지 모드로 메모리 세그먼트에 액세스할 수 있습니다:

읽기 전용 메모리 세그먼트에 액세스할 수 있는 모드 "a", 새 메모리 세그먼트를 생성하는 읽기 전용 액세스 모드 "c", 또는 메모리가 세그먼트가 이미 존재합니다. 읽기 및 쓰기를 위해 열어 보십시오.
모드 "n", 동일한 키가 이미 존재하는 경우 공유 메모리의 안전한 사용을 위해 생성이 실패합니다.

세 번째 매개변수는 메모리 세그먼트의 권한입니다. 여기에 8진수 값을 제공해야 합니다.

네 번째 매개변수는 메모리 세그먼트 크기를 바이트 단위로 제공합니다. 사용되는 공유 메모리 세그먼트는 고정 길이이므로 데이터를 저장하고 읽을 때 데이터의 길이를 계산해야 합니다. 그렇지 않으면 쓰기가 실패하거나 null 값을 읽을 수 있습니다. .

이 함수는 이 공유 메모리 세그먼트에서 작동하기 위해 다른 함수에서 사용할 수 있는 ID 번호를 반환한다는 점에 유의하세요. 이 ID는 파라미터로 전달되는 시스템 ID와 달리 공유 메모리 액세스 ID입니다. 두 가지를 혼동하지 않도록 주의하세요. 실패하면 shmop_open은 FALSE를 반환합니다. 메모리 블록을 생성할 때 키 매개변수에 변수 대신 상수를 사용하는 것이 좋습니다. 그렇지 않으면 메모리 누수가 발생할 수 있습니다.

shmop_write(메모리 세그먼트에 데이터 쓰기)

이 함수는 fwrite 함수와 유사하며, 두 가지 매개 변수, 즉 개방형 스트림 리소스(fopen에서 반환됨)와 쓰려는 데이터를 사용합니다. shmop_write 함수도 이 작업을 수행합니다.

첫 번째 매개변수는 shmop_open이 반환한 ID로, 작업 중인 공유 메모리 블록을 식별합니다. 두 번째 매개변수는 저장하려는 데이터이고, 마지막 세 번째 매개변수는 쓰기를 시작하려는 위치입니다. 기본적으로 쓰기 시작 위치를 나타내기 위해 항상 0을 사용합니다. 이 함수는 실패 시 FALSE를 반환하고 성공 시 기록된 바이트 수를 반환합니다.

shmop_read(메모리 세그먼트에서 데이터 읽기)

공유 메모리 세그먼트에서 데이터를 읽는 것은 매우 간단합니다. 필요한 것은 열린 메모리 세그먼트와 shmop_read 함수뿐입니다. 이 함수는 일부 매개변수를 허용하며 fread처럼 작동합니다.

여기서 매개변수에 주의하세요. shmop_read 함수는 우리가 이미 알고 있는 shmop_open이 반환한 ID를 허용하지만 다른 두 매개변수도 허용합니다. 두 번째 매개변수는 읽으려는 메모리 세그먼트의 위치이고, 세 번째 매개변수는 읽으려는 바이트 수입니다. 두 번째 매개변수는 항상 0이 될 수 있으며 이는 데이터의 시작을 나타냅니다. 그러나 세 번째 매개변수는 읽고자 하는 바이트 수를 모르기 때문에 문제가 될 수 있습니다.

이것은 오픈 스트림 리소스(fopen에 의해 반환됨)와 스트림에서 읽으려는 바이트 수라는 두 가지 매개 변수를 사용하는 fread 함수에서 수행하는 작업과 매우 유사합니다. 파일을 완전히 읽으려면 파일 크기 함수(파일의 바이트 수를 반환)를 사용하십시오.

shmop_size (메모리 세그먼트 데이터의 실제 크기를 반환)

예를 들어 길이가 100바이트인 메모리 공간을 열었지만 저장된 데이터의 실제 길이는 90에 불과한 경우 반환되는 값은 다음과 같습니다. shmop_size를 사용하면 90입니다.

shmop_delete(메모리 세그먼트 삭제)

이 함수는 삭제하려는 공유 메모리 ID라는 하나의 매개변수만 허용하며 실제로 메모리 세그먼트를 삭제하지는 않습니다. 다른 프로세스가 공유 메모리 세그먼트를 사용하는 동안에는 공유 메모리 세그먼트를 삭제할 수 없기 때문에 메모리 세그먼트를 삭제 대상으로 표시합니다. shmop_delete 함수는 메모리 세그먼트를 삭제하도록 표시하여 다른 프로세스가 이를 열지 못하게 합니다. 이를 삭제하려면 해당 메모리 세그먼트를 닫아야 합니다. 메모리 블록을 생성할 때 키 매개변수에 변수 대신 상수를 사용하는 것이 좋습니다. 그렇지 않으면 메모리 누수가 발생할 수 있습니다.

shmop_close(메모리 세그먼트 닫기)

我们在对内存段进行读取和写入,但完成操作后,我们必须从它解除,这非常类似于处理文件时的 fclose 函数。打开包含一个文件的流并在其中读取或写入数据后,我们必须关闭它,否则将发生锁定。

简单测试结果查看

我是在LNMP环境下操作的,如果你也和我一样,在执行完简单的操作之后,可以使用linux命令查看一下地址和占用大小

# ipcs -m
[root@bogon ~]# ipcs -m

------ Shared Memory Segments --------
key  shmid  owner  perms  bytes  nattch  status  
0x00000000 0       gdm    600     393216   2   dest   
0x00000000 32769    gdm    600     393216   2   dest        
0x4337b101 884750   nobody   644     1024    0

命令说明

key :共享内存的唯一的key值,共享内存通过该key来判断你读取的是哪一块内存。 owner:创建该共享内存块的用户 bytes:该内存块的大小 status:当前状态,如:dest,即将删除等。

项目实际应用小案例

/**
 * 将领技能
 */
class Generalskill_model extends CI_Model {<!-- -->

 private $_memory_key = 0x4337b001; //共享内存地址key
 private $_memory_size = 1048576;  //开辟共享内存大小 //最好根据实际数据长度大小定义。

 public function __construct() {<!-- -->
 parent::__construct();
 }

 public function get_skill_list() {<!-- -->
 $data = [];
 $shmid = @shmop_open($this->_memory_key, &#39;a&#39;, 0644, $this->_memory_size); 

 if ($shmid === FALSE) {<!-- -->
  $shmid = @shmop_open($this->_memory_key, &#39;c&#39;, 0644, $this->_memory_size); 
  $data = $this->return_skill_list();
  shmop_write($shmid, json_encode($data), 0); 
  @shmop_close($shmid);

  return $data;
 }
 $data = json_decode(preg_replace(&#39;/[\x00-\x1F\x80-\x9F]/u&#39;, &#39;&#39;, trim(shmop_read($shmid, 0, $this->_memory_size))), true);
 @shmop_close($shmid);
 return $data; 

 }

 public function return_skill_list() {<!-- --> //这里是一个超大的数组,其实就是把这个数组json化,然后存入共享内存段。 其实可以用redis等其他缓存...这里我就是为了不用redis等其他nosql才用的shmop
 return array (
=> 
 array (&#39;id&#39; => &#39;1&#39;,&#39;animation&#39; => &#39;13&#39;,&#39;skill_type&#39; => &#39;1&#39;,&#39;power_type&#39; => &#39;1&#39;,&#39;site&#39; => &#39;1&#39;,&#39;type&#39; => &#39;1&#39;,&#39;paramete&#39; => &#39;0&#39;,&#39;paramete2&#39; => &#39;0&#39;,&#39;paramete3&#39; => &#39;0&#39;,&#39;chance&#39; => &#39;0&#39;,&#39;ratio&#39; => &#39;1&#39;,
 ),
=> 
 array (&#39;id&#39; => &#39;2&#39;,&#39;animation&#39; => &#39;3&#39;,&#39;skill_type&#39; => &#39;2&#39;,&#39;power_type&#39; => &#39;1&#39;,&#39;site&#39; => &#39;1&#39;,&#39;type&#39; => &#39;1&#39;,&#39;paramete&#39; => &#39;0&#39;,&#39;paramete2&#39; => &#39;0&#39;,&#39;paramete3&#39; => &#39;0&#39;,&#39;chance&#39; => &#39;0&#39;,&#39;ratio&#39; => &#39;2&#39;,
 ),..........................................

当然你要考虑的是,如果数据更新的话,那么内存段也要删除,并且更新数据…通过shmop_delete可以删除 。这就需要你们自己根据项目应用来考虑了

还有就是这篇文章只是为了简单的读,并没有出现复杂的读写,否则可能会出现进程互斥等意想不到的冲突如果复杂,那么就可以考虑信号量了。

위 내용은 PHP가 공유 메모리를 생성하여 로드를 줄이는 방법에 대한 자세한 예의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
이 기사는 jb51.net에서 복제됩니다. 침해가 있는 경우 admin@php.cn으로 문의하시기 바랍니다. 삭제