在Unix/Linux下为了保证在同一时间内某个程序只有一个实例在运行,
即防止二重启动的发生,采用的方法除了用锁定文件之外,还可以用该程
序在系统运行时,在进程表中的信息来进行控制(ps命令看到的内容)。
方法有两种:
第一种是:用进程的command字符串进行防止二重启动的控制
实现步骤:
(1)定义应用程序的名的字符串变量
char EXE_NAME[255]=" app ";
(2)可以用exec或popen去执行ps命令,我这里用popen为例
为了使用popen需要引入#include <unistd.h>;
char PS_CMD[256];
char str[256];
FILE fd;
int bRun = 0;
strcpy(PS_CMD, "ps -ef|grep ");
strcat(PS_CMD, EXE_NAME);
if((fd = popen(PS_CMD, "r")) == NULL) {
printf("call popen failed\n");
return;
} else {
while(fgets(str, 255, fd) != NULL) {
printf("%s\n",str);
bRun = 1;
break;
}
}
if( bRun == 1) {
/* 已经有一个实例在运行 */
exit(0);
}
/* 执行程序的功能 */
这种方法的缺点:
(1)如果其他程序的主执行文件名和要执行的程序相同,并正在执行,那么
本来要执行的程序可以被执行,但却不能执行;
(2)如果本程序是用全路径,或程序名比较长,用ps命令时,输出结果可能会被截短,这样可能会在grep时,被能匹配到要执行文件名,而错误判断这个程序还没有运行,但实际上已经运行了;
第二种是:用进行的PID进行防止二重启动的控制
(1)定一个放置本程序执行的PID的文件
char PID_FILE[255] = "APP.PID";
(2)如果APP.PID文件不存在,则本程序没有运行,创建该文件,并把当前运行程序的PID写入该文件
pid_t pid = -1;
pid = getpid();
/* 把pid写入APP.PID文件 */
/* 执行程序的功能 */
执行(4)
(3)如果APP.PID文件存在,则从APP.PID文件中读取PID值和系统进程表中正在运行进程的PID进行比较,
[
/* 从APP.PID文件中读取PID值,放到PID_Value中 */
strcpy(PS_CMD, "ps -ef|awk '{print $2}'|grep ");
strcat(PS_CMD, PID_Value);
/* 在进程表中查找PID_Value值 */
if((fd = popen(PS_CMD, "r")) == NULL) {
printf("call popen failed\n");
return;
} else {
while(fgets(str, 255, fd) != NULL) {
printf("%s\n",str);
bRun = 1;
break;
}
}
]
如果在进程表中不存在,则表示本程序未被运行,执行(2);
如果在进程表中存在,则执行(4)
(4)退出程序。
我认为第二种方法比较好,这也是一些服务程序用来控制多次启动的常用方法。