解决crond脚本执行并发冲突问题

2024-05-14 03:38

本文主要是介绍解决crond脚本执行并发冲突问题,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

解决crond脚本执行并发冲突问题

问题

在crond配置脚本执行后,经常会出现重复执行的情况。如下面的脚本:

1
2
3
<?php
sleep(70);
?>

如果,在crond中的配置项如下:

1
* /1 * * * * root php . /test .php

就出出现,之前执行的test.php还未结束,新的test.php又被执行。如下:

1
2
3
4
5
[hailong@vhost ~]$ ps aux | grep tt.php
56667     5280  0.0  0.0 103388  4432 pts /2    T    08:06   0:00 vim . /test .php
root      5455  0.0  0.1 225288  8812 ?        Ssl  08:08   0:00 /usr/bin/php /home/hailong/test .php
root      5665  5.0  0.1 225288  8748 ?        Ssl  08:09   0:00 /usr/bin/php /home/hailong/test .php
56667     5675  0.0  0.0  69460   852 pts /2    S+   08:09   0:00 grep tt.php

进程一直堆积的话,可能会把系统资源给耗尽,导致系统宕机。

解决方案

解决方案有两种。各有利弊。

第一种,代码中控制并发

这种方法,就是对代码进行改造。增加是否有进程执行的判断。如下面的代码:

1
2
3
4
5
6
7
8
9
10
11
12
<?php 
$lockfile = '/tmp/mytest.lock'
   
if ( file_exists ( $lockfile )){ 
     exit (); 
}
file_put_contents ( $lockfile , date ( "Y-m-d H:i:s" ));
   
sleep(70);
unlink( $lockfile ); 
?>

代码逻辑很简单,这里就不再解释。这种判断文件是否不存在的方式,会有一个问题。那就是有可能程序未执行到最后,也就是没有删除之前创建的mytest.lock文件。这会导致,之后程序将不能正常执行。

为了解决这个问题,我们对代码进行下改造。不再使用文件是否存在的判断,而是判断进程是否存在。代码修改后如下:

1
2
3
4
5
6
$fp = popen( "ps aux | grep 'test.php' | wc -l" , "r" );
$proc_num = fgets ( $fp );
if ( $proc_num > 3) {
     exit ;
}
sleep(70);

这种方式有一个弊端,就是ps命令要写的精确。避免把不是执行test.php脚本的进程也统计到。如:
我们通过vim打开test.php文件。就会导致上面命令误统计。
另外,$proc_num 为什么要大于3而不是大于1,大家可以想想。如果,想不明白,可以加我微信 1798159444。

第二种,使用linux的flock命令

flock命令提供了文件锁的功能。命令参数如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
[hailong@vhost ~]$ flock -h
flock (util-linux 2.13-pre7)
Usage: flock [-sxun][-w #] fd#
        flock [-sxon][-w #] file [-c] command...
   -s  --shared     Get a shared lock
   -x  --exclusive  Get an exclusive lock
   -u  --unlock     Remove a lock
   -n  --nonblock   Fail rather than wait
   -w  --timeout    Wait for a limited amount of time
   -o  --close      Close file descriptor before running command
   -c  -- command    Run a single command string through the shell
   -h  --help       Display this text
   -V  --version    Display version

使用flock控制并发冲突,我们的crond配置如下:

1
* /1 * * * * root flock -xn /tmp/mytest .lock -c 'php ./test.php'

其实,使用flock也是有坑的。坑的详细信息,请查看crond中使用flock命令的坑

这篇关于解决crond脚本执行并发冲突问题的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/987651

相关文章

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

mysql出现ERROR 2003 (HY000): Can‘t connect to MySQL server on ‘localhost‘ (10061)的解决方法

《mysql出现ERROR2003(HY000):Can‘tconnecttoMySQLserveron‘localhost‘(10061)的解决方法》本文主要介绍了mysql出现... 目录前言:第一步:第二步:第三步:总结:前言:当你想通过命令窗口想打开mysql时候发现提http://www.cpp

SpringBoot启动报错的11个高频问题排查与解决终极指南

《SpringBoot启动报错的11个高频问题排查与解决终极指南》这篇文章主要为大家详细介绍了SpringBoot启动报错的11个高频问题的排查与解决,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一... 目录1. 依赖冲突:NoSuchMethodError 的终极解法2. Bean注入失败:No qu

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

springboot报错Invalid bound statement (not found)的解决

《springboot报错Invalidboundstatement(notfound)的解决》本文主要介绍了springboot报错Invalidboundstatement(not... 目录一. 问题描述二.解决问题三. 添加配置项 四.其他的解决方案4.1 Mapper 接口与 XML 文件不匹配

MySQL新增字段后Java实体未更新的潜在问题与解决方案

《MySQL新增字段后Java实体未更新的潜在问题与解决方案》在Java+MySQL的开发中,我们通常使用ORM框架来映射数据库表与Java对象,但有时候,数据库表结构变更(如新增字段)后,开发人员可... 目录引言1. 问题背景:数据库与 Java 实体不同步1.1 常见场景1.2 示例代码2. 不同操作

Python中ModuleNotFoundError: No module named ‘timm’的错误解决

《Python中ModuleNotFoundError:Nomodulenamed‘timm’的错误解决》本文主要介绍了Python中ModuleNotFoundError:Nomodulen... 目录一、引言二、错误原因分析三、解决办法1.安装timm模块2. 检查python环境3. 解决安装路径问题

如何解决mysql出现Incorrect string value for column ‘表项‘ at row 1错误问题

《如何解决mysql出现Incorrectstringvalueforcolumn‘表项‘atrow1错误问题》:本文主要介绍如何解决mysql出现Incorrectstringv... 目录mysql出现Incorrect string value for column ‘表项‘ at row 1错误报错