一、 进程关系
1、进程组
1.
每个进程都会有一个进程组,进程组是多个进程的集合,多个进程组组成一个会话。
查看进程组的id,其中每个进程组都有一个组长,进程组的id就是该进程组长的id号。
该进程组是否存在取决于进程组中是否有进程,与进程组长是否结束无关。
#include <sys/types.h>
#include <unistd.h>
int setpgid(pid_t pid, pid_t pgid);
pid_t getpgid(pid_t pid);
pid_t getpgrp(void); /* POSIX.1 version */
pid_t getpgrp(pid_t pid); /* BSD version */
int setpgrp(void); /* System V version */
int setpgrp(pid_t pid, pid_t pgid); /* BSD version */
2、会话
1.
会话是由多个进程组
组成的
3、父子进程
二、守护进程的引入
1、进程查看命令ps
(1)
ps -ajx
偏向显示各种有关的ID号
(2)ps -aux
偏向显示进程各种占用资源
2、向进程发送信号指令kill
(1)kill -信号编号 进程ID,向一个进程发送一个信号
(2)kill -9 xxx,将向xxx这个进程发送9号信号,也就是要结束进程
3、何谓守护进程
(1) daemon,表示守护进程,简称为d(进程名后面带d的基本就是守护进程)
(2) 长期运行(一般是开机运行直到关机时关闭)
(3) 与控制台脱离(普通进程都和运行该进程的控制台相绑定,表现为如果终端被强制关闭了则这个终端中运行的所有进程都被会关闭,背后的问题还在于会话
)
(4) 服务器(Server),服务器程序就是一个一直在运行的程序,可以给我们提供某种服务(譬如nfs服务器给我们提供nfs通信方式),当我们程序需要这种服务时我们可以调用服务器程序(和服务器程序通信以得到服务器程序的帮助)来进程这种服务操作。服务器程序一般都实现为守护进程。
4、常见守护进程
(1)
syslogd
,系统日志守护进程,提供syslog功能。
(2)cron
,cron进程用来实现操作系统的时间管理,linux中实现定时执行程序的功能就要用到cron。
三、编写简单守护进程
任何一个进程都可以将自己实现成守护进程
create_daemon函数要素
(
1
) 子进程等待父进程退出
(2
) 子进程使用setsid创建新的会话期,脱离控制台
(3
) 调用chdir将当前工作目录设置为/
(4
)umask设置为0以取消任何文件权限屏蔽
(5
)关闭所有文件描述符
(6
)将0、1、2定位到/dev/null
将标准输入、输出、错误 定位到null
设备 就会导致该进程传递参数和输出信息都定位到/dev/null设备中(垃圾堆中)
/dev/null
为字符设备,相当于一个回收站。
代码部分
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
void create_daemon(void);
int main(void)
{
create_daemon();
while (1)
{
printf("I am running.\n");
sleep(1);
}
return 0;
}
// 函数作用就是把调用该函数的进程变成一个守护进程
void create_daemon(void)
{
pid_t pid = 0;
pid = fork();
if (pid < 0)
{
perror("fork");
exit(-1);
}
if (pid > 0)
{
exit(0); // 父进程直接退出
}
// 执行到这里就是子进程
// setsid将当前进程设置为一个新的会话期session,目的就是让当前进程
// 脱离控制台。
pid = setsid();
if (pid < 0)
{
perror("setsid");
exit(-1);
}
// 将当前进程工作目录设置为根目录
chdir("/");
// umask设置为0确保将来进程有最大的文件操作权限
umask(0);
// 关闭所有文件描述符
// 先要获取当前系统中所允许打开的最大文件描述符数目
int cnt = sysconf(_SC_OPEN_MAX);
int i = 0;
for (i=0; i<cnt; i++)
{
close(i);
}
open("/dev/null", O_RDWR);//0 将标准输入与null设备绑定
open("/dev/null", O_RDWR);//1 将标准输出与null设备绑定
open("/dev/null", O_RDWR);//2 将标准错误与null设备绑定
}