UNIX高级编程【深入浅出】 进程关系(下)

目录

进程关系

会话session

进程组 process group

进程process

进程组组长

创建会话 setsid(2)

交互进程和守护进程

如何构建守护进程

构建守护进程函数 daemon(3)

守护进程实例


进程关系
  1. 会话session
    1. 打开一个终端就是创建一个会话
    2. 会话就是一个容器,承载进程组
    3. 会话标识 SID == 第一个进程组的ID
    4. 获取会话ID:getsid(2);
    5. 设置会话:setsid(2);
  2. 进程组 process group
    1. 进程组也是一个容器
    2. 承载进程的
    3. 进程组的标识 PGID == 进程组内第一个进程的PID
    4. 前台进程组和后台进程组
      1. 一个会话内可以由一个前台进程组和若干个后台进程组
      2. 前台进程组就是容纳交互进程的
    5. 获取:getpgid(2); / getpgrp(2);
    6. 设置:setpgid(2);
  3. 进程process
    1. 也是容器,承载线程的
  4. 进程组组长
    1. 会话内第一个进程组内的第一个进程就是所在进程组的组长也是所在会话的组长。
      1. 组长的PID == PGID == SID
  5. 创建会话 setsid(2)
    1. 功能是创建一个新的会话,前提是非组长进程调用
    2. 调用进程将为称为新的会话内,新的进程组内唯一的进程
    3. SID == PID == PGID
    4. 脱离控制终端
  6. 交互进程和守护进程
    1. 交互进程是占用终端的进程,可以在终端交互的
    2. 守护进程就是脱离控制终端的,并且这个进程是所在进程组组长,所在会话组长。
  7. 如何构建守护进程
    1. fork(2)子进程,父进程终止
    2. 子进程调用setsid(2)
    3. PID == PGID == SID
    4. 不占用控制终端
    5. 0, 1, 2对应标准输入标准输出和标准错误输出文件,这三个文件都是与终端交互的,所以守护进程应该重定向0, 1, 2,建议重定向到"/dev/null"
    6. 每一个进程都有文件屏蔽字,如果0这个进程占用终端,那么我们在终端通过umask命令是可以修改bash的文件屏蔽字的,进而在终端下执行的进程也就是bash的子进程也就随着修改了文件屏蔽字。
      1. 守护进程不占用终端,所以文件屏蔽字就不能修改了
      2. 建议将守护进程的文件屏蔽字提前设置为0
    7. 每个进程都有工作目录,守护进程不占用终端,建议将工作目录切换到"/",chdir(3)
  8. 构建守护进程函数 daemon(3)
    1. int daemon(int nochdir, int noclose);
      1. nochdir是0则将进程的工作目录切换到/
      2. noclose是0则将0, 1, 2重定向到/dev/null
  9. 程序写日志
    1. 日志
      1. openlog(3)
      2. syslog(3)
      3. closelog(3)
      4. linux中有一个专门写日志的进程system logger
      5. syslog提交给system logger,默认写"/var/log/syslog"
  10. 单实例守护进程
    1. 同时只能执行一次
    2. flock(2)给文件加锁,锁的类型有共享锁,互斥锁(同时只能一个进程得到)
  11. 守护进程实例
// Date:2023.08.30 16:10:56
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <sys/stat.h>
#include <time.h>
#include <string.h>
#include <fcntl.h>
#include <syslog.h>
#include <errno.h>
#include <sys/file.h>

#define FLNAME "/tmp/out"  // 写入时间的文件
#define DEVNULL "/dev/null"
#define RUNDAEMON "/var/run/daemon.pid" // 写入id的文件 [sudo]
#define FILENAME "03protect" // 当前文件名
#define BUFFSIZE 128

/* 守护进程 例[写入时间到文件 转后台] */
char *get_time(char *buf);
int mydaemon(void);
int alread_running(void);
int main(void)
{
	int i = 0;
	char buf[BUFFSIZE] = {};
	FILE *fp = NULL;

	// 创建连接 
	openlog(FILENAME, LOG_PID, LOG_DAEMON);
	// 守护进程
	if(-1 == mydaemon())
		exit(1);
	if(NULL == (fp = fopen(FLNAME, "w"))){
		//perror("fopen()");
		// 错误输出到日志
		syslog(LOG_ERR, "[%d]fopen():%s", __LINE__, strerror(errno));
		return 1;
	}

	// 单实例
	if(-1 == alread_running()){
		syslog(LOG_ERR, "the process is already running");
		exit(1);
	}
	
	while(1){
		/* 程序正常运行日志 [Debug] */ 
		if(++i == 5){
			syslog(LOG_INFO, "%s is running", buf);
			i = 0;
		}
		fputs(get_time(buf), fp);
		fputs("\n", fp);
		fflush(NULL);    // 全缓存, 需要刷新
	}

	return 0;
}
char *get_time(char *a)
{
	char *buf = a;
	time_t tn;
	struct tm *tim = NULL;
	time(&tn);	
	memset(buf, '\0', BUFFSIZE);
	if(NULL == (tim = localtime(&tn)))
		return NULL;
	strftime(buf, BUFFSIZE, "%y-%m-%d %H:%M:%S", tim);
	sleep(1);
	return a;
}

// 守护进程---》精灵进程 daemon
int mydaemon(void)
{
	pid_t pid;
	int fd;
	/* 创建子进程 */
	if(-1 == (pid = fork())){
		perror("fork()");
		return -1;
	}

	/* 关闭父进程 */
	if(pid > 0)
		exit(0);

	/* child 作为会话 */
	if(-1 == setsid()){
		perror("setsid()");
		return -1;
	}

	// PID == PGID PID == SID 脱落控制终端
	umask(0);  // 给予更多权限
	chdir("/");// 将当前目录 --> 根目录
	
	// 0 1 2 重定向[关闭不需要的文件描述符]
	if(-1 == (fd = open(DEVNULL, O_RDWR))){
		perror("open()");
		return -1;
	}
	dup2(fd, 0);
	dup2(fd, 1);
	dup2(fd, 2);
	if(fd > 2)
		close(fd);

	return 0;
}

// 单实例 [加锁,只执行一次]
int alread_running(void)
{
	int fd;
	char buf[BUFFSIZE] = {};
	// 在/run文件下产生一个pid文件,供管理人员查看id号
	if(-1 == (fd = (open(RUNDAEMON, O_WRONLY | O_CREAT, 0666)))){
		syslog(LOG_ERR, "open():%s", strerror(errno));
		return -1;
	}

	// 加锁
	if(-1 == flock(fd, LOCK_EX | LOCK_NB)){
		syslog(LOG_ERR, "flock():%s", strerror(errno));
		return -1;
	}

	// 调用进程目前还没执行过
	snprintf(buf, BUFFSIZE, "%d", getpid());

	// 清空原文件内容
	ftruncate(fd, 0);
	write(fd, buf, strlen(buf));
	// 文件不要要关闭,否则会影响锁

	return 0;

}
  1. ps -axj | tail 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值