>  기사  >  백엔드 개발  >  PHP 공유 메모리 사용량 및 신호 제어 예시 분석

PHP 공유 메모리 사용량 및 신호 제어 예시 분석

不言
不言원래의
2018-05-09 10:22:451182검색

이 글은 주로 PHP 공유 메모리 사용 분석 및 신호 제어 예제를 소개합니다. 이제는 모든 사람과 공유합니다.

이 글은 PHP 공유 메모리 사용 및 신호를 설명합니다. 제어 예. 참고용으로 모든 사람과 공유하세요. 세부 사항은 다음과 같습니다.

공유 메모리

공유 메모리의 사용은 주로 동일한 시스템의 여러 프로세스에서 일부 데이터를 공유할 수 있도록 하는 것입니다. 여러 php-fpm 프로세스 현재 프로세스 사용량. 이러한 종류의 통신을 프로세스 간 통신(Inter-Process Communication), 줄여서 IPC라고도 합니다.

PHP에 내장된 shmop 확장(공유 메모리 작업)은 공유 메모리 작업을 위한 일련의 기능을 제공합니다(아마도 많은 사람들이 사용하지 않기 때문에 이 문서는 아직 중국어로 번역되지 않았습니다). Linux에서는 이러한 함수가 shm* 시리즈 함수를 호출하여 직접 구현되는 반면, Winodows에서는 시스템 함수를 캡슐화하여 동일한 호출도 구현됩니다.

주요 기능:

shmop_close — 공유 메모리 블록 닫기

shmop_delete — 공유 메모리 블록 삭제

shmop_open — 공유 메모리 블록 생성 또는 열기

shmop_ 읽기 — 공유 메모리 블록에서 읽기 data

shmop_size — 공유 메모리 블록의 크기 가져오기

shmop_write — 공유 메모리 블록에 데이터 쓰기

이와 관련된 매우 중요한 기능도 있습니다: ftok, 파일의 inode 정보를 통해 (*nix에서 stat 또는 ls -i 명령을 통해 보기) IPC에 대한 고유 키를 생성합니다(파일/폴더의 inode는 고유합니다). 이 함수는 Linux에서도 같은 이름의 시스템 함수를 직접 호출하여 구현되며, Windows에서는 여전히 일부 캡슐화가 사용됩니다.

간단한 계산 예:

<?php
# 创建一块共享内存
$shm_key = ftok(__FILE__, &#39;t&#39;);
$shm_id = shmop_open($shm_key, &#39;c&#39;, 0644, 8);
# 读取并写入数据
$count = (int) shmop_read($shm_id, 0, 8) + 1;
shmop_write($shm_id, str_pad($count, 8, &#39;0&#39;, STR_PAD_LEFT), 0);
// echo shmop_read($shm_id, 0, 8);
# 关闭内存块,并不会删除共享内存,只是清除 PHP 的资源
shmop_close($shm_id);

위 코드는 1씩 증가하는 계산을 실행하지 않으며, 데이터는 서로 다른 프로세스 간에 공유됩니다. 즉, 이 메모리를 수동으로 삭제하지 않는 한 이 데이터는 재설정되지 않습니다.

약간의 주의가 필요한 점이 있습니다. shmop_open의 두 번째 매개변수는 플래그이며, fopen의 두 번째 매개변수와 유사하며 해당 값은 다음과 같습니다.

"a" 읽기 전용 액세스;

"c" 메모리 세그먼트가 없으면 생성합니다. 존재하면 읽고 쓸 수 있습니다.

"w"는 읽고 씁니다.

"n"은 새 메모리 세그먼트를 생성합니다. 키가 이미 존재하면 생성이 실패합니다. 이는 공유 메모리의 안전한 사용을 고려하기 위한 것입니다.

또한, 사용되는 공유 메모리 세그먼트는 고정 길이이므로 데이터를 저장하고 읽을 때 데이터의 길이를 계산해야 합니다. 그렇지 않으면 쓰기가 실패하거나 null 값을 읽을 수 있습니다.

신호 제어

위의 데이터를 저장하기 위해 공유 메모리를 사용하므로 여러 프로세스가 동시에 공유 메모리에 데이터를 쓰는지 여부와 충돌을 피해야 하는지 고려해야 합니다. 그렇다면 제어를 위한 세마포어를 도입해야 합니다.

PHP도 유사한 내장 확장인 sysvsem을 제공합니다(이 확장은 Windows 환경에서는 사용할 수 없습니다. 문서에서는 이 확장에 ftok 함수도 포함되어 있지만 실제로는 표준 함수 라이브러리에 ftok가 제공되어 있으므로 Windows available에서도 사용 가능).

세마포어 제어에 대해 이야기하기 전에 또 다른 흥미로운 점에 대해 이야기하겠습니다. 공식 문서를 보면 공유 메모리 작업(shm_*)을 위한 함수도 있다는 것을 알 수 있습니다. 동일한 작성자) ), 하나는 sysvmsg(대기열 메시지)입니다. 기능의 구현은 약간 다르지만 실제로 수행하는 작업은 기본적으로 동일합니다. 위의 shmop 확장과 차이점은 무엇입니까? shmop 소스 코드 아래의 README 파일에는 간단한 설명이 있습니다:

PHP에는 이미 Christian Cartus 32982644a4ae8a6ce575374773bddd52가 작성한 공유 메모리 확장(sysvshm)이 있지만 불행히도 이 확장은 PHP만을 염두에 두고 설계되었습니다. 우리가 염두에 두었던 기본 SHM에는 매우 귀찮은 고급 기능을 제공합니다.

간단히 말하면: sysvshm 확장에서 제공하는 방법은 사용자의 데이터를 그대로 저장하는 것이 아니라 먼저 PHP의 변수 직렬화 기능을 사용하여 처리하는 것입니다. 매개변수를 직렬화한 후 저장합니다. 이는 이러한 방법을 통해 저장된 데이터를 PHP가 아닌 프로세스와 공유할 수 없음을 의미합니다. 그러나 이는 더 풍부한 PHP 데이터 유형을 저장할 수도 있습니다. 위 확장에서 shmop_write는 문자열만 쓸 수 있습니다. 그렇다면 sysvshm이 Windows도 지원하지 않는 이유는 무엇입니까? shm* 계열 함수를 캡슐화하는 tsrm_win32.h의 헤더 파일을 도입하지 않기 때문입니다.

신호 제어 도입 후의 예:

<?php
$id_key = ftok(__FILE__, &#39;t&#39;);
$sem_id = sem_get($id_key);
# 请求信号控制权
if (sem_acquire($sem_id)) {
  $shm_id = shmop_open($id_key, &#39;c&#39;, 0644, 8);
  # 读取并写入数据
  $count = (int) shmop_read($shm_id, 0, 8) + 1;
  shmop_write($shm_id, str_pad($count, 8, &#39;0&#39;, STR_PAD_LEFT), 0);
  // echo shmop_read($shm_id, 0, 8);
  # 关闭内存块
  shmop_close($shm_id);
  # 释放信号
  sem_release($sem_id);
}

그러나 쓰기 충돌을 로컬에서 시뮬레이션하는 것은 실제로 매우 어렵습니다(컴퓨터의 실행 속도를 고려). 로컬 테스트에서 for 루프 연산 사용 시 리소스 닫기를 위해 shmop_close를 사용하지 않으면 공유 메모리를 열 수 없다는 오류 경고가 나타납니다. 이는 이전 작업에서 공유 메모리를 점유하고 아직 해제되지 않았기 때문에 발생할 수 있습니다.

관련 권장 사항:

shmop 시리즈 함수는 PHP 공유 메모리 구현 방식을 사용합니다.

PHP 공유 메모리에서 사용하는 코드에 대한 자세한 소개


위 내용은 PHP 공유 메모리 사용량 및 신호 제어 예시 분석의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명:
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.