PHP 利用文件锁处理高并发
利用flock()函数对文件进行加锁(排它锁),实现并发按序进行。 flock(file,lock,block)有三个参数。 file:已经打开的文件 lock:锁的类型 LOCK_SH:共享锁(读锁) LOCK_EX:独占锁定(排它锁,写锁) LOCK_UN:解锁 LOCK_NB:如果希望在文件锁定时阻塞进程,那么需要加上该参数 block:设置为true的时候,锁定文件时,会阻止其他进程 下面是一个demo:class Order{
/**
* 阻塞模式(后面的进程会一直等待前面的进程执行完毕)
*/
public function createOrder1(){
$file = fopen(__DIR__.'/lock.txt','w+');
//加锁
if(flock($file,LOCK_EX)){
//TODO 执行业务代码
flock($file,LOCK_UN);//解锁
}
//关闭文件
fclose($file);
}
/**
* 非阻塞模式(只要当前文件有锁存在,那么直接返回)
*/
public function createOrder2(){
$file = fopen(__DIR__.'/lock.txt','w+');
//加锁
if(flock($file,LOCK_EX|LOCK_NB)){
//TODO 执行业务代码
flock($file,LOCK_UN);//解锁
}else{
//TODO 执行业务代码 返回系统繁忙等错误提示
}
//关闭文件
fclose($file);
}
}提示:在执行fclose()的时候文件会自动解锁,所以可以省略解锁的代码。
demo:【一】.阻塞模式:(如果其他进程已经加锁文件,当前进程会一直等其他进程解锁文件后继续执行)
<?php
//连接数据库
$con=mysqli_connect("192.168.2.186","root","root","test");
//查询商品数量是否大于0,大于0才能下单,并减少库存
$fp = fopen("lock.txt", "r");
//加锁
if(flock($fp,LOCK_EX))
{
$res=mysqli_fetch_assoc(mysqli_query($con,'SELECT total FROM shop WHERE id=1 LIMIT 1'));
if($res['total']>0){mysqli_query($con,'UPDATE shop SET total=total-1WHERE id=1');}
//执行完成解锁
flock($fp,LOCK_UN);
}
//关闭文件
fclose($fp);
unset($res);
mysqli_close($con);
?>
这种情况若是其他进程已经加锁文件,那么所有进程都会等他执行完并解锁文件后才会执行
【二】.非阻塞模式:(如果其他进程已经加锁文件,当前进程不会等其他进程解锁文件,而是走else)<?php
//连接数据库
$con=mysqli_connect("192.168.2.186","root","root","test");
//查询商品数量是否大于0,大于0才能下单,并减少库存
$fp = fopen("lock.txt", "r");
//加锁
if(flock($fp,LOCK_EX | LOCK_NB))
{
$res=mysqli_fetch_assoc(mysqli_query($con,'SELECT total FROM shop WHERE id=1 LIMIT 1'));
if($res['total']>0){mysqli_query($con,'UPDATE shop SET total=total-1WHERE id=1');}
//执行完成解锁
flock($fp,LOCK_UN);
}else{
echo "locked file failed\n";
}
unset($res);
mysqli_close($con);
?>
这种情况就会直接走else返回提示信息
页:
[1]