linux 只启动一个进程实例

时间:2022-02-17 17:46:29

linux 只启动一个进程实例


====================================================================
我使用的是第一种方法,文件锁。参考《UNIX环境高级编程》13.5,14.3实现
====================================================================

很多应用程序都只应该启动一个运行实例,特别是设计文件IO操作的程序,如果误启动了多个实例,可能会导致混乱的IO,进而导致文件写乱。

常见的做法:

1.生成pid文件
这种做法最常见了,应用于大多数的Linux程序,如apache httpd, mysql。思路是配置一个pid文件,当程序启动时,对pid文件加锁,然后写入本进程的pid,如果锁失败,说明有实例已经启动了。这个方案非常可靠,唯一的不足是需要配置一个pid文件,并且保证文件目录和文件可写。

2.端口抢占
应用于大多数的Linux网络应用。思路是系统保证每个端口的TCP只能有一个进程监听,那么如果程序启动时,监听一个核心的端口,第二个运行的实例就会监听失败,无法启动。这个方案同样很有效,省去了一个额外的配置文件,不足之处是一般只用于带网络的程序。

3.共享内存锁
比较小众的方法,对方案1, 2的优化。方案1中的pid需要保存一个文件,文件就需要配置路径和权限,并且这个文件仅供进程使用,意义不大,所以考虑使用共享内存,通过shmget操作共享内存,然后写入pid,这样就不用生成可见的文件。这个方案只存在一个很小的缺陷,需要配置共享内存的key,并保证不与系统其他应用冲突,一般来说,冲突概率非常小。

4.进程列表检测
对于运维常用的方法。由于运维不一定能控制程序的修改,所以考虑从外部解决。crontab脚本,查询运行的进程数,一旦发现进程数与预期不符,那么killall,重启进程。这个方案是旁路方案,比上面的方式更通用,还可以监视进程数,避免某些子进程core。这个方案没有什么缺陷,如果硬要找一个的话,不同的系统ps命令输出可能不一样,脚本需要考虑移植。