3.4.12.进程关系
(1)无关系
两个进程有独立的进程空间,
(2)父子进程关系
父进程 fork 创建了 子进程
父进程 open 打开了一个文件,父进程 fork, 就会把 打开的文件 继承给 子进程
(3)进程组(group): 由若干进程构成一个进程组
好几个进程 放到一个组,就叫进程 组。
(4)会话(session)会话 :就是进程组的组
好几个进程组 放到一个 组 就叫 会话!!
3.4.13.守护进程的引入
3.4.13.1、进程查看命令ps
(1)ps -ajx 偏向显示各种有关的ID号
(2)ps -aux 偏向显示进程各种占用资源
3.4.13.2、向进程发送信号指令kill
(1)kill -信号编号 进程ID,向一个进程发送一个信号
(2)kill -9 xxx,将向xxx这个进程发送9号信号,也就是要结束进程
信号 9 : 代表 关闭信号
3.4.13.3、何谓守护进程
(1)daemon,表示守护进程,简称为d(进程名后面带d的基本就是守护进程)
(2)长期运行(一般是开机运行直到关机时关闭)
(3)与控制台脱离(普通进程都和运行该进程的控制台(终端)相绑定,表现为如果终端被强制关闭了则这个终端中运行的所有进程都被会关闭,背后的问题还在于会话)
控制台被关闭,但是守护进程不会被关闭
(4)服务器(Server),服务器程序就是一个一直在运行的程序,可以给我们提供某种服务(譬如nfs服务器给我们提供nfs通信方式),当我们程序需要这种服务时我们可以调用服务器程序(和服务器程序通信以得到服务器程序的帮助)来进程这种服务操作。服务器程序一般都实现为守护进程。
守护进程: 就是独立于控制台,长期运行! 守护进程 是应用程序,不是驱动和内核
3.4.13.4、常见守护进程
(1)syslogd,系统日志守护进程,提供syslog 系统日志功能。
(2)cron,cron进程用来实现操作系统的时间管理,linux中实现定时执行程序的功能就要用到cron。
3.4.14.编写简单守护进程
3.4.14.1、任何一个进程都可以将自己实现成守护进程
3.4.14.2、create_daemon函数要素
(1)子进程等待父进程退出
(2)子进程使用setsid创建新的会话期,脱离控制台
(3)调用chdir将当前工作目录设置为/
(4)umask设置为0以取消任何文件权限屏蔽
(5)关闭所有文件描述符
/*先要 获取当前系统中 所允许 打开的 最大文件描述符数目 */
#include <unistd.h>
long sysconf(int name);
OPEN_MAX - _SC_OPEN_MAX
The maximum number of files that a process can have open at any
time. Must not be less than _POSIX_OPEN_MAX (20).
OPEN_MAX:任何一个进程再任何一个时间可以打开的 最大个数 的文件
sysconf(_SC_OPEN_MAX);
(6)将0、1、2定位到/dev/null
代码:
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
void create_demon(void);
int main(void)
{
create_demon();
while(1)
{
printf("I am runint \n");
sleep(1);
}
return 0;
}
// 函数作用:就是把调用该函数的进程 变成 一个守护进程
void create_demon(void)
{
pid_t pid = -1;
pid = fork();
/* pid 小于 0,代表 fork 出错了 */
if(pid < 0)
{
perror("fork");
exit(-1);
}
/* pid 大于 0,代表 父进程 */
if(pid > 0)
{
exit(0);/* 父进程 直接退出 */
}
/* 能执行 到这里一定是 子进程: 不用 在 判断 pid == 0 了 */
pid = setsid(); /* setsid 将当前进程 设置为 一个新的会话器 session, 目的就是让当前进程 脱离控制台 */
if(pid < 0)
{
perror("setsid");
exit(-1);
}
chdir("/");/* 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 标准输入 */
open("/dev/null", O_RDWR); /*打开 文件描述 符 1 标准输出*/
open("/dev/null", O_RDWR); /*打开 文件描述 符 2 标准错误*/
}
运行结果:
代码 来自朱老师 物联网
扩展:
本来想着 把标准 输出 2.txt:
open("1.txt", O_RDWR); /*打开 文件描述 符 0 标准输入 */
open("2.txt", O_RDWR); /*打开 文件描述 符 1 标准输出*/
chdir("/winshare/AppNet/4.file");/* chdir : 设置当前进程的目录为 根目录 */
这是 更改 的这 3 处 地方:
代码:
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdlib.h>
void create_demon(void);
int main(void)
{
printf("创建 \n");
create_demon();
printf("创建完毕 \n");
while(1)
{
printf("I am runint \n");
sleep(1);
}
return 0;
}
// 函数作用:就是把调用该函数的进程 变成 一个守护进程
void create_demon(void)
{
pid_t pid = -1;
pid = fork();
/* pid 小于 0,代表 fork 出错了 */
if(pid < 0)
{
perror("fork");
exit(-1);
}
/* pid 大于 0,代表 父进程 */
if(pid > 0)
{
exit(0);/* 父进程 直接退出 */
}
/* 能执行 到这里一定是 子进程: 不用 在 判断 pid == 0 了 */
pid = setsid(); /* setsid 将当前进程 设置为 一个新的会话器 session, 目的就是让当前进程 脱离控制台 */
if(pid < 0)
{
perror("setsid");
exit(-1);
}
chdir("/winshare/AppNet/4.file");/* chdir : 设置当前进程的目录为 根目录 */
umask(0);/*umask 设置为 0 ,确保将来进程有最大的文件操作权限 */
/*关闭所有 文件描述 符 */
int cnt = sysconf(_SC_OPEN_MAX); /*先要 获取当前系统中 所允许 打开的 最大文件描述符数目 */
int i = 0;
for(i=0; i<cnt; i++)
{
close(i);
}
open("1.txt", O_RDWR); /*打开 文件描述 符 0 标准输入 */
open("2.txt", O_RDWR); /*打开 文件描述 符 1 标准输出*/
//open("3.txt", O_RDWR); /*打开 文件描述 符 2 标准错误*/
printf("你好吗 \n");
}
运行结果: