利用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-1 WHERE 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-1 WHERE id=1');}
- //执行完成解锁
- flock($fp,LOCK_UN);
- }else{
- echo "locked file failed\n";
- }
- unset($res);
- mysqli_close($con);
- ?>
- 这种情况就会直接走else返回提示信息
复制代码
|