Home  >  Article  >  Backend Development  >  Method of using locking to implement code grabbing function under concurrency based on PHP

Method of using locking to implement code grabbing function under concurrency based on PHP

墨辰丷
墨辰丷Original
2018-06-01 15:44:181322browse

This article is based on the PHP language and uses locking to implement the code grabbing function under concurrent conditions. The code grabbing is open during a specific period of time and does not allow the open codes to be repeated. The introduction in this article is very detailed. Friends who need it can refer to the

Requirements : Code grabbing function

Requirements:

1. Code grabbing is only open during a specific time period;

2.The codes released in each time period are limited;

3. Each code is not allowed to be repeated;

Implementation:

1. Implementation without considering concurrency:

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

The above code satisfies the current situation by default is the opening time, and the codes are not repeated;

Process without considering concurrency:

1) Select to query the number of verification codes issued by the current database;

2 ) If there are still places, generate a verification code, insert it into the database, and return the verification code to the client;

 3) If it is full; then return a prompt that there are no places;

2, Implemented under concurrent conditions:

Then take a look at the results obtained by the above code under concurrent conditions:

To test concurrency, you can use apache benchmark to test. apache benchmark is the performance of the HTTP SERVER under APACHE. For the evaluation tool, enter the bin directory of apche through cmd and call it through the ab command, such as: ab -c concurrent number -n total visits url

The code is as follows:


cb -c 100 -n 100 http://localhost/php_mulit.php

In this way, 100 users and colleagues compete for 1 quota. When querying, each user finds that there is still one quota, then It will generate a verification code, insert it into the database, and return the verification code; this will cause too many verification codes to be sent. In fact, after running this command, the database has 13 more records instead of one.

How to avoid this happening?

You can lock the process from judgment to insertion by adding an exclusive lock to ensure that only one process is running in this judgment process at any time. The implementation is as follows:

//生成码
function get_code($len){
$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;);
$CHAR_ARR_LEN = count($CHAR_ARR) - 1;
$code = &#39;&#39;;
while(--$len > 0){ $code .= $CHAR_ARR[rand(0,$CHAR_ARR_LEN)]; }
return $code;
}
$pdo = new PDO(&#39;mysql:host=localhost;dbname=ci_test&#39;,&#39;root&#39;,&#39;root&#39;);
$fp = fopen(&#39;lock.txt&#39;,&#39;r&#39;);
//通过排他锁 锁定该过程
if(flock($fp,LOCK_EX)){
//查询当前时间已发放验证码数量
$code_num_rs = $pdo->query("SELECT COUNT(*) as sum FROM code_test");
$code_num_arr = $code_num_rs->fetch(PDO::FETCH_ASSOC);
$code_num = $code_num_arr[&#39;sum&#39;];
if($code_num < 1){
sleep(2);
$code = get_code(6);
var_dump( $pdo->query("INSERT INTO code_test (code,create_time) VALUES (&#39;$code&#39;,".time().")") );
}
flock($fp,LOCK_UN);
fclose($fp);
}

Lock the process through flock function.

For more flock information, please refer to the php manual: http://php.net/manual/zh/function.flock.php

Run again

The code is as follows:


cb -c 100 -n 100 http://localhost/php_mulit.php

Only one record is added to the database, ensuring that the data is correct under concurrent conditions.

Summary: The above is the entire content of this article, I hope it will be helpful to everyone's study.

Related recommendations:

php Detailed examples of the difference between die() and exit()

# #phpmailer implements the method of binding mailbox

phpUsage of image processing function imagecopyresampled

The above is the detailed content of Method of using locking to implement code grabbing function under concurrency based on PHP. For more information, please follow other related articles on the PHP Chinese website!

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn