php flock 超时,php flock失效问题

这两天给自己的业余项目写了一个方法,用来避免crontab调度的PHP脚本并发执行。

做法

一般通过使用文件锁flock方法,令相同的PHP脚本采用非阻塞锁同一个磁盘文件,如果文件被占用则会报错,从而可以脚本立即退出。

现象

但实践中发现,在controller文件中直接flock是可以实现的,当把flock的逻辑封装到其他文件的一个函数中后就失效了。

原因

调试了半天,突然想起来以前就遇到过这个神坑。。

错误代码如下:

PHP

class Crontab

{

/**

* 确保任务没有并发执行

*/

public static function isRunning() {

global $argv;

$ident = [];

foreach ($argv as $idx => $value) {

$ident[] = $idx . '=' . urlencode($value);

}

$ident = md5(implode('&', $ident));

$lockDir = \Yii::getAlias('@app/runtime/crontab/');

@mkdir($lockDir, 0755, true);

$file_lock = fopen($lockDir . $ident, 'w+');

$wouldBlock = 0;

flock($file_lock, LOCK_EX | LOCK_NB, $wouldBlock);

return $wouldBlock;

}

}

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

classCrontab

{

/**

* 确保任务没有并发执行

*/

publicstaticfunctionisRunning(){

global$argv;

$ident=[];

foreach($argvas$idx=>$value){

$ident[]=$idx.'='.urlencode($value);

}

$ident=md5(implode('&',$ident));

$lockDir=\Yii::getAlias('@app/runtime/crontab/');

@mkdir($lockDir,0755,true);

$file_lock=fopen($lockDir.$ident,'w+');

$wouldBlock=0;

flock($file_lock,LOCK_EX|LOCK_NB,$wouldBlock);

return$wouldBlock;

}

}

根据命令行参数生成唯一hash值,代表该PHP任务。

创建锁文件,执行flock非阻塞锁,返回wouldBlock标识锁是否已被占用。

我在脚本入口调用了Crontab::isRunning()方法,发现并发启动脚本后,总是能获得锁。

错误原因是:isRunning()方法退出后,$file_lock没有继续使用,被PHP垃圾回收,$fp文件句柄关闭导致锁自动释放。

解决

PHP

class Crontab

{

/**

* 保存起来避免被php作为垃圾回收

* @var null

*/

static $file_lock = null;

/**

* 确保任务没有并发执行

*/

public static function isRunning() {

global $argv;

$ident = [];

foreach ($argv as $idx => $value) {

$ident[] = $idx . '=' . urlencode($value);

}

$ident = md5(implode('&', $ident));

$lockDir = \Yii::getAlias('@app/runtime/crontab/');

@mkdir($lockDir, 0755, true);

self::$file_lock = fopen($lockDir . $ident, 'w+');

$wouldBlock = 0;

flock(self::$file_lock, LOCK_EX | LOCK_NB, $wouldBlock);

return $wouldBlock;

}

}

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

classCrontab

{

/**

* 保存起来避免被php作为垃圾回收

* @var null

*/

static$file_lock=null;

/**

* 确保任务没有并发执行

*/

publicstaticfunctionisRunning(){

global$argv;

$ident=[];

foreach($argvas$idx=>$value){

$ident[]=$idx.'='.urlencode($value);

}

$ident=md5(implode('&',$ident));

$lockDir=\Yii::getAlias('@app/runtime/crontab/');

@mkdir($lockDir,0755,true);

self::$file_lock=fopen($lockDir.$ident,'w+');

$wouldBlock=0;

flock(self::$file_lock,LOCK_EX|LOCK_NB,$wouldBlock);

return$wouldBlock;

}

}

确保在整个PHP生命期内,文件句柄都不会被释放即可,所以保存在类静态成员变量里。

如果文章帮助您解决了工作难题,您可以帮我点击屏幕上的任意广告,或者赞助少量费用来支持我的持续创作,谢谢~

c68972f84f7c4f47f59a1f69f0608e10.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值