찾다
백엔드 개발PHP 튜토리얼PHP의 파일 잠금, 뮤텍스 잠금 및 읽기-쓰기 잠금에 대한 자세한 설명

이 글에서는 주로 PHP 프로그램의 파일 잠금, 뮤텍스 잠금 및 읽기-쓰기 잠금 사용 기술을 소개하며, 동기화 모듈의 사용 예를 중점적으로 다루며 필요한 친구들이 참고할 수 있습니다. 그것이 모두에게 도움이 되기를 바랍니다.

파일 잠금
전체 이름은 책에 언급된 권고 파일 잠금입니다. 이러한 유형의 잠금은 비교적 일반적입니다. 예를 들어, mysql과 php-fpm이 시작된 후에는 프로세스 ID를 기록하는 pid 파일이 있습니다.

이 잠금을 사용하면 프로세스가 반복적으로 실행되는 것을 방지할 수 있습니다. 예를 들어 crontab을 사용하면 1분마다 하나의 작업이 실행되도록 제한되지만 프로세스 잠금을 사용하여 문제를 해결하지 않으면 이 프로세스가 1분 이상 실행될 수 있습니다. 충돌이 발생하여 두 프로세스를 함께 실행하면 문제가 발생합니다.

PID 파일 잠금 사용의 또 다른 장점은 프로세스가 중지 또는 다시 시작 신호를 자체적으로 보내는 것이 편리하다는 것입니다. 예를 들어 php-fpm을 다시 시작하는 명령은

kill -USR2 `cat /usr/local/php/var/run/php-fpm.pid`
pid 파일에 기록된 프로세스에 USR2 신호를 보냅니다. 신호는 프로세스 통신에 속하며 별도로 처리됩니다.

PHP의 인터페이스는 매우 다양하며 문서는 비교적 자세합니다. 먼저 정의를 살펴보겠습니다. bool Flock (resource $handle , int $Operation [, int &$wouldblock ] ).

  • $handle은 일반적으로 fopen()에 의해 생성되는 리소스인 파일 시스템 포인터입니다. . 이는 Flock을 사용하려면 파일을 열어야 함을 의미합니다.

  • $Operation은 작업 유형입니다.

  • &$wouldblock 잠금이 차단되면 이 변수는 1로 설정됩니다.

이 기능은 기본적으로 차단된다는 점에 유의하세요. 작업에 대한 비트마스크 LOCK_NB를 내려서 테스트해 보세요.


$pid_file = "/tmp/process.pid";
$pid = posix_getpid();
$fp = fopen($pid_file, 'w+');
if(flock($fp, LOCK_EX | LOCK_NB)){
  echo "got the lock \n";
  ftruncate($fp, 0);   // truncate file
  fwrite($fp, $pid);
  fflush($fp);      // flush output before releasing the lock
  sleep(300); // long running process
  flock($fp, LOCK_UN);  // 释放锁定
} else {
  echo "Cannot get pid lock. The process is already up \n";
}
fclose($fp);

process.php로 저장하고 php process.php &를 실행한 다음 php process.php를 다시 실행하면 오류 메시지가 표시됩니다. Flock에는 공유 잠금인 LOCK_SH도 있습니다.

뮤텍스 잠금 및 읽기-쓰기 잠금
동기화 모듈의 뮤텍스:
Mutex는 복합어, 상호 배제입니다. pecl을 사용하여 동기화 모듈을 설치하고 pecl install sync를 설치합니다. 문서의 SyncMutex에는 잠금 및 잠금 해제 두 가지 메서드만 있습니다. 바로 코드 테스트로 이동하겠습니다. 저는 IDE에서 작성하지 않았으므로 cs가 매우 추악합니다. 무시하십시오.


$mutex = new SyncMutex("UniqueName");

for($i=0; $i<2; $i++){
  $pid = pcntl_fork();
  if($pid <0){
    die("fork failed");
  }elseif ($pid>0){
    echo "parent process \n";
  }else{
    echo "child process {$i} is born. \n";
    obtainLock($mutex, $i);
  }
}

while (pcntl_waitpid(0, $status) != -1) { 
  $status = pcntl_wexitstatus($status); 
  echo "Child $status completed\n"; 
}

function obtainLock ($mutex, $i){
  echo "process {$i} is getting the mutex \n";
  $res = $mutex->lock(200);
  sleep(1);
  if (!$res){
    echo "process {$i} unable to lock mutex. \n";
  }else{
    echo "process {$i} successfully got the mutex \n";
    $mutex->unlock();
  }
  exit();
}

mutex.php로 저장하고 php mutex.php를 실행하면 출력은


parent process 
parent process 
child process 1 is born. 
process 1 is getting the mutex 
child process 0 is born. 
process 0 is getting the mutex 
process 1 successfully got the mutex 
Child 0 completed
process 0 unable to lock mutex. 
Child 0 completed

여기서 하위 프로세스 0과 1이 반드시 앞에 올 필요는 없습니다. 그러나 자물쇠를 얻지 못하는 사람은 항상 있습니다. 여기서 SyncMutex::lock(int $millisecond)의 매개변수는 차단 기간을 나타내는 밀리초이고, -1은 무제한 차단을 의미합니다.

동기화 모듈의 읽기-쓰기 잠금:
SyncReaderWriter의 메서드는 유사하며, readlock, readunlock, writelock, writeunlock이 쌍으로 나타날 수 있습니다. 작성된 테스트 코드는 없습니다. 자물쇠를 교체하십시오.

동기화 모듈의 이벤트:
golang의 Cond와 더 비슷하고, wait() 블록과 fire()는 Event에 의해 차단된 프로세스를 깨웁니다. Cond를 소개하는 좋은 기사가 있습니다. Cond는 고정된 잠금 사용법임을 알 수 있습니다. SyncEvent도 마찬가지입니다.
PHP 문서의 예는 fire() 메소드가 웹 애플리케이션에서 사용되는 것으로 보입니다.

for($i=0; $i<3; $i++){
  $pid = pcntl_fork();
  if($pid <0){
    die("fork failed");
  }elseif ($pid>0){
    //echo "parent process \n";
  }else{
    echo "child process {$i} is born. \n";
    switch ($i) {
    case 0:
      wait();
      break;
    case 1:
      wait();
      break;
    case 2:
      sleep(1);
      fire();
      break;
    }
  }
}

while (pcntl_waitpid(0, $status) != -1) { 
  $status = pcntl_wexitstatus($status); 
  echo "Child $status completed\n"; 
}

function wait(){
  $event = new SyncEvent("UniqueName");
  echo "before waiting. \n";
  $event->wait();
  echo "after waiting. \n";
  exit();
}

function fire(){
  $event = new SyncEvent("UniqueName");
  $event->fire();
  exit();
}

의 테스트 코드 여기에는 의도적으로 fire()가 하나 덜 작성되어 프로그램이 차단됩니다. 이는 fire()가 한 번에 하나의 프로세스만 깨운다는 것을 증명합니다.

pthreads 모듈
뮤텍스 잠금 및 잠금 해제:

기능:


pthread_mutex_lock (mutex) 
pthread_mutex_trylock (mutex) 
pthread_mutex_unlock (mutex)

사용법:

뮤텍스가 다른 스레드에 의해 잠긴 경우 스레드는 pthread_mutex_lock() 함수를 사용하여 지정된 뮤텍스 변수를 잠급니다. , 이 호출은 뮤텍스가 잠금 해제될 때까지 스레드를 차단합니다.
pthread_mutex_trylock()은 뮤텍스를 잠그려고 시도합니다. 그러나 뮤텍스가 이미 잠겨 있으면 루틴은 "busy" 오류 코드와 함께 즉시 반환됩니다. 이 루틴은 pthread_mutex_trylock()에서 유용할 수 있습니다.

  Mutex. 뮤텍스가 잠겨 있으면 프로그램은 사용 중인 오류 값과 함께 즉시 반환됩니다. 이 기능은 우선순위 변경 시 교착상태를 방지하는 데 유용합니다. 스레드는 pthread_mutex_unlock()을 사용하여 자신이 차지하는 뮤텍스를 잠금 해제할 수 있습니다. 한 스레드가 보호된 데이터의 사용을 완료하고 다른 스레드가 보호된 데이터에 대해 작업하기 위해 뮤텍스를 얻으려고 할 때 이 함수를 호출할 수 있습니다. 다음 상황이 발생하면 오류가 발생합니다. 뮤텍스가 잠금 해제되었습니다. 뮤텍스가 다른 스레드에 의해 점유되었습니다. 이는 참여 스레드의 "신사 합의"입니다. 코드를 작성할 때 뮤텍스를 올바르게 잠그고 잠금 해제해야 합니다.

    Q: 동일한 잠긴 뮤텍스를 기다리는 스레드가 여러 개 있습니다. 뮤텍스가 잠금 해제되면 어떤 스레드가 뮤텍스를 가장 먼저 잠그나요?
  • A: 스레드가 우선 순위 예약 메커니즘을 사용하지 않는 한 시스템 스케줄러에 의해 스레드가 할당되며 뮤텍스를 가장 먼저 잠글 스레드는 무작위입니다.

  • #include<stdlib.h> 
    #include<stdio.h> 
    #include<unistd.h> 
    #include<pthread.h> 
    
    typedef struct ct_sum 
    { 
      int sum; 
      pthread_mutex_t lock; 
    }ct_sum; 
    
    void * add1(void *cnt) 
    {    
      pthread_mutex_lock(&(((ct_sum*)cnt)->lock)); 
      for(int i=0; i < 50; i++) 
      {
        (*(ct_sum*)cnt).sum += i;   
      } 
      pthread_mutex_unlock(&(((ct_sum*)cnt)->lock)); 
      pthread_exit(NULL); 
      return 0; 
    } 
    void * add2(void *cnt) 
    {    
      pthread_mutex_lock(&(((ct_sum*)cnt)->lock)); 
      for(int i=50; i<101; i++) 
      {  
         (*(ct_sum*)cnt).sum += i;  
      } 
      pthread_mutex_unlock(&(((ct_sum*)cnt)->lock)); 
      pthread_exit(NULL); 
      return 0; 
    } 
     
    int main(void) 
    {
      pthread_t ptid1, ptid2; 
      ct_sum cnt; 
      pthread_mutex_init(&(cnt.lock), NULL); 
      cnt.sum=0; 
     
      pthread_create(&ptid1, NULL, add1, &cnt); 
      pthread_create(&ptid2, NULL, add2, &cnt); 
      
      pthread_join(ptid1,NULL); 
      pthread_join(ptid2,NULL);
    
      printf("sum %d\n", cnt.sum);
      pthread_mutex_destroy(&(cnt.lock)); 
    
      return 0; 
    }

    信号量
    sync模块中的信号量:
    SyncSemaphore文档中显示,它和Mutex的不同之处,在于Semaphore一次可以被多个进程(或线程)得到,而Mutex一次只能被一个得到。所以在SyncSemaphore的构造函数中,有一个参数指定信号量可以被多少进程得到。
    public SyncSemaphore::__construct ([ string $name [, integer $initialval [, bool $autounlock ]]] ) 就是这个$initialval (initial value)


    $lock = new SyncSemaphore("UniqueName", 2);
    
    for($i=0; $i<2; $i++){
      $pid = pcntl_fork();
      if($pid <0){
        die("fork failed");
      }elseif ($pid>0){
        echo "parent process \n";
      }else{
        echo "child process {$i} is born. \n";
        obtainLock($lock, $i);
      }
    }
    
    while (pcntl_waitpid(0, $status) != -1) { 
      $status = pcntl_wexitstatus($status); 
      echo "Child $status completed\n"; 
    }
    
    function obtainLock ($lock, $i){
      echo "process {$i} is getting the lock \n";
      $res = $lock->lock(200);
      sleep(1);
      if (!$res){
        echo "process {$i} unable to lock lock. \n";
      }else{
        echo "process {$i} successfully got the lock \n";
        $lock->unlock();
      }
      exit();
    }

    这时候两个进程都能得到锁。

    • sysvsem模块中的信号量

    • sem_get 创建信号量

    • sem_remove 删除信号量(一般不用)

    • sem_acquire 请求得到信号量

    • sem_release 释放信号量。和 sem_acquire 成对使用。


    $key = ftok(&#39;/tmp&#39;, &#39;c&#39;);
    
    $sem = sem_get($key);
    
    for($i=0; $i<2; $i++){
      $pid = pcntl_fork();
      if($pid <0){
        die("fork failed");
      }elseif ($pid>0){
        //echo "parent process \n";
      }else{
        echo "child process {$i} is born. \n";
        obtainLock($sem, $i);
      }
    }
    
    while (pcntl_waitpid(0, $status) != -1) { 
      $status = pcntl_wexitstatus($status); 
      echo "Child $status completed\n"; 
    }
    sem_remove($sem); // finally remove the sem
    
    function obtainLock ($sem, $i){
      echo "process {$i} is getting the sem \n";
      $res = sem_acquire($sem, true);
      sleep(1);
      if (!$res){
        echo "process {$i} unable to get sem. \n";
      }else{
        echo "process {$i} successfully got the sem \n";
        sem_release($sem);
      }
      exit();
    }

    这里有一个问题,sem_acquire()第二个参数$nowait默认为false,阻塞。我设为了true,如果得到锁失败,那么后面的sem_release会报警告 PHP Warning:  sem_release(): SysV semaphore 4 (key 0x63000081) is not currently acquired in /home/jason/sysvsem.php on line 33, 所以这里的release操作必须放在得到锁的情况下执行,前面的几个例子中没有这个问题,没得到锁执行release也不会报错。当然最好还是成对出现,确保得到锁的情况下再release。
    此外,ftok这个方法的参数有必要说明下,第一个 必须是existing, accessable的文件, 一般使用项目中的文件,第二个是单字符字符串。返回一个int。

    输出为


    parent process 
    parent process 
    child process 1 is born. 
    process 1 is getting the mutex 
    child process 0 is born. 
    process 0 is getting the mutex 
    process 1 successfully got the mutex 
    Child 0 completed
    process 0 unable to lock mutex. 
    Child 0 completed

    相关推荐:

    简单介绍PHP 文件锁与进程锁

    php 文件读取系列方法详解

    简单谈谈 php 文件锁

위 내용은 PHP의 파일 잠금, 뮤텍스 잠금 및 읽기-쓰기 잠금에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
PHP 세션에 어떤 데이터를 저장할 수 있습니까?PHP 세션에 어떤 데이터를 저장할 수 있습니까?May 02, 2025 am 12:17 AM

phpsessionscanstorestrings, 숫자, 배열 및 객체 1.Strings : TextDatalikeUsernames.2.numbers : integorfloatsforcounters.3.arrays : listslikeshoppingcarts.4.objects : complexStructuresThatareserialized.

PHP 세션을 어떻게 시작합니까?PHP 세션을 어떻게 시작합니까?May 02, 2025 am 12:16 AM

tostartAphPessession, us

세션 재생이란 무엇이며 보안을 어떻게 개선합니까?세션 재생이란 무엇이며 보안을 어떻게 개선합니까?May 02, 2025 am 12:15 AM

세션 재생은 세션 고정 공격의 경우 사용자가 민감한 작업을 수행 할 때 새 세션 ID를 생성하고 이전 ID를 무효화하는 것을 말합니다. 구현 단계에는 다음이 포함됩니다. 1. 민감한 작업 감지, 2. 새 세션 ID 생성, 3. 오래된 세션 ID 파괴, 4. 사용자 측 세션 정보 업데이트.

PHP 세션을 사용할 때 몇 가지 성능 고려 사항은 무엇입니까?PHP 세션을 사용할 때 몇 가지 성능 고려 사항은 무엇입니까?May 02, 2025 am 12:11 AM

PHP 세션은 응용 프로그램 성능에 큰 영향을 미칩니다. 최적화 방법은 다음과 같습니다. 1. 데이터베이스를 사용하여 세션 데이터를 저장하여 응답 속도를 향상시킵니다. 2. 세션 데이터 사용을 줄이고 필요한 정보 만 저장하십시오. 3. 비 차단 세션 프로세서를 사용하여 동시성 기능을 향상시킵니다. 4. 사용자 경험과 서버 부담의 균형을 맞추기 위해 세션 만료 시간을 조정하십시오. 5. 영구 세션을 사용하여 데이터 읽기 및 쓰기 시간의 수를 줄입니다.

PHP 세션은 쿠키와 어떻게 다릅니 까?PHP 세션은 쿠키와 어떻게 다릅니 까?May 02, 2025 am 12:03 AM

phpsessionsareser-side, whilecookiesareclient-side.1) sessions stessoredataontheserver, andhandlargerdata.2) cookiesstoredataonthecure, andlimitedinsize.usesessionsforsensitivestataondcookiesfornon-sensistive, client-sensation.

PHP는 사용자 세션을 어떻게 식별합니까?PHP는 사용자 세션을 어떻게 식별합니까?May 01, 2025 am 12:23 AM

phpidifiesauser의 sssessionusessessioncookiesandssessionids.1) whensession_start () iscalled, phpgeneratesauniquessessionStoredInacookienamedPhpsSessIdonSeuser 'sbrowser.2) thisidallowsphptoretrievessessionDataTromServer.

PHP 세션을 확보하기위한 모범 사례는 무엇입니까?PHP 세션을 확보하기위한 모범 사례는 무엇입니까?May 01, 2025 am 12:22 AM

PHP 세션의 보안은 다음 측정을 통해 달성 할 수 있습니다. 1. Session_REGENEREAT_ID ()를 사용하여 사용자가 로그인하거나 중요한 작업 일 때 세션 ID를 재생합니다. 2. HTTPS 프로토콜을 통해 전송 세션 ID를 암호화합니다. 3. 세션 _save_path ()를 사용하여 세션 데이터를 저장하고 권한을 올바르게 설정할 보안 디렉토리를 지정하십시오.

PHP 세션 파일은 기본적으로 어디에 저장됩니까?PHP 세션 파일은 기본적으로 어디에 저장됩니까?May 01, 2025 am 12:15 AM

phpsessionfilesarestoredInTheRectorySpecifiedBysession.save_path, 일반적으로/tmponunix-likesystemsorc : \ windows \ temponwindows.tocustomizethis : 1) austession_save_path () toSetacustomDirectory, verlyTeCustory-swritation;

See all articles

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

SecList

SecList

SecLists는 최고의 보안 테스터의 동반자입니다. 보안 평가 시 자주 사용되는 다양한 유형의 목록을 한 곳에 모아 놓은 것입니다. SecLists는 보안 테스터에게 필요할 수 있는 모든 목록을 편리하게 제공하여 보안 테스트를 더욱 효율적이고 생산적으로 만드는 데 도움이 됩니다. 목록 유형에는 사용자 이름, 비밀번호, URL, 퍼징 페이로드, 민감한 데이터 패턴, 웹 셸 등이 포함됩니다. 테스터는 이 저장소를 새로운 테스트 시스템으로 간단히 가져올 수 있으며 필요한 모든 유형의 목록에 액세스할 수 있습니다.

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

DVWA

DVWA

DVWA(Damn Vulnerable Web App)는 매우 취약한 PHP/MySQL 웹 애플리케이션입니다. 주요 목표는 보안 전문가가 법적 환경에서 자신의 기술과 도구를 테스트하고, 웹 개발자가 웹 응용 프로그램 보안 프로세스를 더 잘 이해할 수 있도록 돕고, 교사/학생이 교실 환경 웹 응용 프로그램에서 가르치고 배울 수 있도록 돕는 것입니다. 보안. DVWA의 목표는 다양한 난이도의 간단하고 간단한 인터페이스를 통해 가장 일반적인 웹 취약점 중 일부를 연습하는 것입니다. 이 소프트웨어는

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

MinGW - Windows용 미니멀리스트 GNU

MinGW - Windows용 미니멀리스트 GNU

이 프로젝트는 osdn.net/projects/mingw로 마이그레이션되는 중입니다. 계속해서 그곳에서 우리를 팔로우할 수 있습니다. MinGW: GCC(GNU Compiler Collection)의 기본 Windows 포트로, 기본 Windows 애플리케이션을 구축하기 위한 무료 배포 가능 가져오기 라이브러리 및 헤더 파일로 C99 기능을 지원하는 MSVC 런타임에 대한 확장이 포함되어 있습니다. 모든 MinGW 소프트웨어는 64비트 Windows 플랫폼에서 실행될 수 있습니다.