当前位置:   article > 正文

会话、守护进程与线程_会话线程

会话线程

Linux系统编程——进程组与会话

进程组和会话

进程组包含多个进程,进程组ID=第一个进程的ID(亦即组长进程ID)。

可使用 kill -SIGKILL -进程组ID 来杀死整个进程组的所有进程。

会话则是多个进程组。

会话、进程组、进程的关系图:

在这里插入图片描述

​ 举例:

在这里插入图片描述

​ 然后利用ps -ajx查看进程信息

(ps ajx 命令查看系统中的进程。参数 a 表示不仅列当前用户的进程,也列出所有其他用户的进程,参数 x 表示不仅列有控制终端的进程,也列出所有无控制终端的进程,参数 j 表示列出与作业控制相关的信息.)
  • 1

在这里插入图片描述

创建会话
  1. 调用进程不能是进程组组长,该进程会变成新会话的首进程(session header);
  2. 该进程成为一个新进程组的组长进程
  3. 需要有root权限(ubuntu不用);
  4. 新会话会丢弃原有的控制终端,该会话没有控制终端;
  5. 该调用进程是组长进程,会出错返回;
  6. 建立新会话时,先调用fork,然后终止父进程,再用子进程调用setsid();
getsid()函数
/*
成功:返回调用进程的会话ID
失败:-1,并设置errno
*/
pid_t getsid(pid_t pid);//传入参数pid为0时,表示查看看当前进程的session ID
  • 1
  • 2
  • 3
  • 4
  • 5
setsid()函数
/*
成功:返回调用进程的会话ID
失败:-1,并设置errno
*/
pid_t setsid(void);//创建一个会话,并以自己的进程ID设置进程组和新会话的ID
  • 1
  • 2
  • 3
  • 4
  • 5
代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <pthread.h>

void sys_err(const char* str) {
	perror(str);
	exit(1);
}

int main(int argc, char* argv[]) {
	pid_t pid;

	pid = fork();
	if(pid < 0 ){	//fork失败
		sys_err("fork error");
	} else if( pid == 0 ) {	//子进程
		printf("I'm a child proc:%d\n", getpid());
		printf("I'm a child, my group proc:%d\n", getpgid(pid));
		printf("I'm a child, my session proc:%d\n", getsid(pid));

		sleep(10);

		setsid();//利用该子进程创建新的会话

		printf("session created:\n");

		printf("I'm a child proc:%d\n", getpid());
		printf("I'm a child, my group proc:%d\n", getpgid(pid));
		printf("I'm a child, my session proc:%d\n", getsid(pid));
		
	} else {	//父进程
		sleep(20);//父进程等待子进程结束后再执行
		printf("I'm a parent proc:%d\n", getpid());
	}

	return 0;	
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40

守护进程

Daemon进程,是Linux中的后台服务进程,通常独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件,一般不与用户直接交互。一般采用以d结尾的名字。

不受用户登录注销影响。

创建守护进程模型:
  1. fork子进程,让父进程终止;
  2. 子进程调用setsid()创建新会话;
  3. 通常根据需要,改变工作目录位置,利用chdir()函数;
  4. 通常根据需要,重设umask文件权限掩码,防止继承的文件创建屏蔽字拒绝某些权限,以增加守护进程的灵活性;
  5. 通常根据需要,关闭/重定向文件描述符;
  6. 开始守护进程的核心工作业务逻辑,一般是while(1){……}。
代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <errno.h>
#include <pthread.h>
#include <fcntl.h> //STDIN_FILENO etc.
#include <sys/stat.h>

void sys_err(const char* str) {
	perror(str);
	exit(1);
}

int main(int argc, char* argv[]) {
	pid_t pid;
	int ret,fd;

	pid = fork();	//创建子进程
	if(pid > 0) {	//让父进程终止
		exit(0);
	}

	pid = setsid();	//创建会话
	if( pid == -1) {	//处理创建会话失败的情形
		sys_err("setsid error");
	}

	ret = chdir("/home/");	//改变工作目录
	if( ret == -1) {	//处理改变工作目录失败的情形
		sys_err("chdir error");
	}

	umask(0022);	//改变文件访问权限的掩码,在此设置为755

	close(STDIN_FILENO);	//关闭标准输入文件描述符
	fd = open("/dev/null",O_RDWR);	// fd-->0
	if( fd == -1) {	//处理改变工作目录失败的情形
		sys_err("open error");
	}

	dup2(fd, STDOUT_FILENO);	//重定向标准输出
	dup2(fd, STDERR_FILENO);	//重定向标准错误输出

	while(1);	//模拟守护进程核心业务逻辑

	return 0;	
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48

执行后,可看到daemon为守护进程,进程号、进程组和会话号均相同。

在这里插入图片描述

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Monodyee/article/detail/236376
推荐阅读
相关标签
  

闽ICP备14008679号