当我们使用 C/C++ 开发一个守护进程或者一个服务端程序的时候,有时需要将该程序变成单进程,防止重复打开 socket 端口或者提供重复的服务。如果某个进程同时有多个实例运行,那么每个实例都可能尝试打开同一个端口或执行某个预定的操作,于是造成该操作的重复执行,这很可能导致出错。
在 POSIX 系统中可以使用文件和记录锁机制来达成单实例进程设计。
文件和记录锁机制为一种方法提供了基础,该方法保证一个守护进程只有一个副本在运行。如果每一个守护进程创建一个固定名字的文件,并在该文件的整体上加一把写锁,那么只允许创建一把这样的写锁。在此之后创建写锁的尝试都会失败,这向后续进程副本指明已有一个副本正在运行。
文件和记录锁提供了一种方便的互斥机制。如果守护进程在一个文件的整体上得到一把写锁,那么在该守护进程终止时,这把锁将被自动删除。这就简化了复原所需的处理,去除了对以前的守护进程实例需要进行清理的有关操作。
——《UNIX 环境高级编程(第 3 版)》
#include // included for `fprintf`
#include // included for `open`
#include // included for `errno`
#include // inlcuded for `strerror`
#include // inlcuded for `fcntl`
#include // included for `exit`
#include // included for `flock`#define PIDFILE "/var/run/hello.pid"int isRunning(const char *pidfile) {FILE *fp = NULL;int fd, otherpid;if ((-1 == (fd = open(pidfile, O_RDWR | O_CREAT, 0666))) ||(NULL == (fp = fdopen(fd, "r+")))) {fprintf(stderr, "can't open or create %s: %s\n",pidfile, strerror(errno));return -1;}if (flock(fd, LOCK_EX | LOCK_NB) < 0) {int save_errno = errno;fscanf(fp, "%d", &otherpid);fprintf(stderr, "can't lock %s, otherpid may be %d: %s(%d)\n",pidfile, otherpid, strerror(save_errno), save_errno);return 1;}(void)fcntl(fd, F_SETFD, 1);rewind(fp);fprintf(fp, "%d\n", getpid());fflush(fp);(void)ftruncate(fileno(fp), ftell(fp));return 0;
}int main(int argc, char *argv[]) {if (0 != isRunning(PIDFILE)) {exit(1);}while (true) {pause();}return 0;
}
每个进程都将尝试创建一个文件并将自己的 PID 号写入到文件中,如果文件已经上锁,那么 flock
函数就会失败并返回 EAGAIN 错误码。
#include int flock(int fd, int operation);
其中 fd
是要操作的文件描述符。operation
可以是以下的值:
如果另一个进程持有不兼容的锁,则对 flock()
的调用可能会阻塞。要发出非阻塞请求,请将 LOCK_NB(通过“或”运算)与上述任何 operation
一起使用。
单个文件不能同时具有共享和独占锁。
在命令行执行 gcc
进行编译:
gcc -Wall -Werror -o hello hello.c
因为 pid 文件指定在 /var/run
目录,所以需要 root 权限执行,否则会报 can't open or create /var/run/hello.pid: Permission denied
错误。
sudo ./hello
进程启动后,可以看到创建了 /var/run/hello.pid
文件。当再启动一个终端执行 hello
进程时将会得到报错信息:
sudo ./hello
can't lock /var/run/hello.pid, otherpid may be 6227: Resource temporarily unavailable(11)
因为已经有另一个进程实例在运行,所以通过文件和记录锁机制就能够达到单实例进程的目的。
[1] 《UNIX 环境高级编程(第 3 版)》13.5 单实例守护进程
[2] QNX 7.1 交叉编译 cron