>백엔드 개발 >PHP 튜토리얼 >PHP는 잠금을 통해 동시 스냅업 기능을 구현합니다.

PHP는 잠금을 통해 동시 스냅업 기능을 구현합니다.

*文
*文원래의
2017-12-29 18:09:042464검색

이 글은 PHP 언어를 기반으로 하며 잠금을 사용하여 동시 조건에서 긴급 구매 기능을 구현합니다. 긴급 구매는 특정 기간 동안 열려 있으며 오픈 코드의 중복을 허용하지 않습니다. 이 글은 매우 자세하게 설명되어 있으니 필요하신 분은 참고하시기 바랍니다. 그것이 모두에게 도움이 되기를 바랍니다.

요구 사항: 코드 잡기 기능

1. 코드 잡기는 특정 기간에만 열려 있습니다.

2. 각 기간에 출시되는 코드는 제한되어 있습니다. 중복은 허용되지 않습니다.

구현:

1. 동시성을 고려하지 않은 구현:

<p style="margin-top: 5px; margin-bottom: 5px;">function get_code($len){<br/>$CHAR_ARR = array(&#39;1&#39;,&#39;2&#39;,&#39;3&#39;,&#39;4&#39;,&#39;5&#39;,&#39;6&#39;,&#39;7&#39;,&#39;8&#39;,&#39;9&#39;,&#39;A&#39;,&#39;B&#39;,&#39;C&#39;,&#39;D&#39;,&#39;E&#39;,&#39;F&#39;,&#39;G&#39;,&#39;H&#39;,&#39;I&#39;,&#39;J&#39;,&#39;K&#39;,&#39;L&#39;,&#39;M&#39;,&#39;N&#39;,&#39;O&#39;,&#39;P&#39;,&#39;Q&#39;,&#39;X&#39;,&#39;Y&#39;,&#39;Z&#39;,&#39;W&#39;,&#39;S&#39;,&#39;R&#39;,&#39;T&#39;);<br/>$CHAR_ARR_LEN = count($CHAR_ARR) - 1;<br/>$code = &#39;&#39;;<br/>while(--$len > 0){ $code .= $CHAR_ARR[rand(0,$CHAR_ARR_LEN)]; }<br/>return $code;<br/>}<br/>$pdo = new PDO(&#39;mysql:host=localhost;dbname=ci_test&#39;,&#39;root&#39;,&#39;root&#39;);<br/>//查询当前时间已发放验证码数量<br/>$code_num_rs = $pdo->query("SELECT COUNT(*) as sum FROM code_test");<br/>$code_num_arr = $code_num_rs->fetch(PDO::FETCH_ASSOC);<br/>$code_num = $code_num_arr[&#39;sum&#39;];<br/>if($code_num < 1){<br>   sleep(2); //暂停2秒<br/>$code = get_code(6);<br/>var_dump( $pdo->query("INSERT INTO code_test (code,create_time) VALUES (&#39;$code&#39;,".time().")") );<br/>}<br/></p>

위 코드는 기본적으로 현재 열기 시간을 만족하며, 코드가 반복되지 않습니다.


 1) 현재 데이터베이스에서 발급된 인증코드 개수를 선택하여 조회합니다.

 2) 아직 장소가 있으면 인증코드를 생성하여 데이터베이스에 삽입한 후 클라이언트에 인증코드를 반환합니다.

 3) 이미 꽉 차 있으면 할당량이 없다는 메시지가 반환됩니다.


2. 동시 조건 하에서 구현:


  그런 다음 동시 조건 하에서 위 코드로 얻은 결과를 살펴보십시오.


  동시성을 테스트하려면 apache 벤치마크를 사용하면 됩니다. apache Benchmark는 APACHE 아래 HTTP SERVER의 성능 평가 도구입니다. cmd를 통해 apche의 bin 디렉터리를 입력하고 ab -c number와 같은 명령을 통해 호출합니다. of concurrency -n 총 방문 횟수 url

<p style="margin-top: 5px; margin-bottom: 5px;">cb -c 100 -n 100 http://localhost/php_mulit.php<br/></p>

 100명의 사용자 동료가 1개의 할당량을 가져가는 방법입니다. 쿼리할 때 각 사용자가 다른 할당량이 있음을 발견하면 확인 코드가 생성되어 데이터베이스에 삽입됩니다. 인증 코드가 반환됩니다. 이로 인해 인증 코드가 너무 많이 발급될 수 있습니다. 실제로 이 명령을 실행한 후 데이터베이스에는 1개가 아닌 13개의 레코드가 더 포함됩니다.


이런 일이 발생하지 않도록 하려면 어떻게 해야 하나요?

  이 판단 과정에서 언제든지 하나의 프로세스만 실행되도록 전용 잠금을 추가하여 판단부터 삽입까지 프로세스를 잠글 수 있습니다. 구현은 다음과 같습니다:

<p style="margin-top: 5px; margin-bottom: 5px;">//生成码<br/>function get_code($len){<br/>$CHAR_ARR = array(&#39;1&#39;,&#39;2&#39;,&#39;3&#39;,&#39;4&#39;,&#39;5&#39;,&#39;6&#39;,&#39;7&#39;,&#39;8&#39;,&#39;9&#39;,&#39;A&#39;,&#39;B&#39;,&#39;C&#39;,&#39;D&#39;,&#39;E&#39;,&#39;F&#39;,&#39;G&#39;,&#39;H&#39;,&#39;I&#39;,&#39;J&#39;,&#39;K&#39;,&#39;L&#39;,&#39;M&#39;,&#39;N&#39;,&#39;O&#39;,&#39;P&#39;,&#39;Q&#39;,&#39;X&#39;,&#39;Y&#39;,&#39;Z&#39;,&#39;W&#39;,&#39;S&#39;,&#39;R&#39;,&#39;T&#39;);<br/>$CHAR_ARR_LEN = count($CHAR_ARR) - 1;<br/>$code = &#39;&#39;;<br/>while(--$len > 0){ $code .= $CHAR_ARR[rand(0,$CHAR_ARR_LEN)]; }<br/>return $code;<br/>}<br/>$pdo = new PDO(&#39;mysql:host=localhost;dbname=ci_test&#39;,&#39;root&#39;,&#39;root&#39;);<br/>$fp = fopen(&#39;lock.txt&#39;,&#39;r&#39;);<br/>//通过排他锁 锁定该过程<br/>if(flock($fp,LOCK_EX)){<br/>//查询当前时间已发放验证码数量<br/>$code_num_rs = $pdo->query("SELECT COUNT(*) as sum FROM code_test");<br/>$code_num_arr = $code_num_rs->fetch(PDO::FETCH_ASSOC);<br/>$code_num = $code_num_arr[&#39;sum&#39;];<br/>if($code_num < 1){<br/>sleep(2);<br/>$code = get_code(6);<br/>var_dump( $pdo->query("INSERT INTO code_test (code,create_time) VALUES (&#39;$code&#39;,".time().")") );<br/>}<br/>flock($fp,LOCK_UN);<br/>fclose($fp);<br/>}<br/></p>

플록 기능을 통해 프로세스를 잠급니다.

다시 실행

<p style="margin-top: 5px; margin-bottom: 5px;">cb -c 100 -n 100 http://localhost/php_mulit.php<br/></p>

하나의 레코드만 데이터베이스에 추가되어 동시성에서 데이터가 올바른지 확인합니다.

관련 권장 사항:

브라우저가 종료된 후에도 PHP가 계속 실행되는지에 대한 심층 탐색


PHP가 온라인 주소록을 구현하는 방법의 예

PHP 방법 대용량 파일을 효율적으로 읽을 수 있습니다. 파일 비교 예시

위 내용은 PHP는 잠금을 통해 동시 스냅업 기능을 구현합니다.의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

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