linux下防止后台程序重复加载

来源:互联网 发布:smarty二维数组遍历 编辑:程序博客网 时间:2024/05/01 02:22

在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)退出程序。

我认为第二种方法比较好,这也是一些服务程序用来控制多次启动的常用方法。

 

++:

特定s h e l l变量
$# 传递到脚本的参数个数
$* 以一个单字符串显示所有向脚本传递的参数。与位置变量不同,此选项参数可超过 9个
$$ 脚本运行的当前进程I D号
$! 后台运行的最后一个进程的进程I D号
$@ 与$ #相同,但是使用时加引号,并在引号中返回每个参数
$- 显示s h e l l使用的当前选项,与s e t命令功能相同
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。

[code]
#!/bin/sh

PIDLock=/tmp/.program.lock
if [ -r $PIDLock ];then
  echo "stop..."
  cat $PIDLock | read pid
  kill -0 $pid > /dev/null 2>&1
  if [ $? -eq 0 ]; then
  echo "The same program is running now."
  exit 1
  fi
else
  echo "start... "
  touch $PIDLock
  echo "888" > $PIDLock
fi
echo $$ > $PIDLock


[/code]