一.进程(续)
1.waitpid函数
#include <sys/types.h>
#include <sys/wait.h>
//
pid_t waitpid(pid_t pid,int *status,int options);
pid不同值的含义:
- pid > 0:等待指定进程结束
- pid = 0:等待同一个进程组中的任何进程
- pid = -1:等待任何子进程
- pid < -1:等待指定进程组中的任意子进程,其进程组的ID为pid的绝对值
options值的含义:
- 0:同wait函数,会阻塞父进程,等待子进程结束
- WNOHANG:没有任何已经结束的子进程,则立即返回
- WUNTRACED:如果子进程暂停则函数马上返回,并不会检测子进程的结束状态
返回值的含义:
- 当正常回收时,返回子进程的进程号
- 当设置了WNOHANG,还有子进程在运行,且没有退出,则返回0;父进程的子进程全部退出返回 -1 ;当大于0则有一个子进程退出
- 当调用时出错,则返回-1
2.创建多进程
int i = 0;
for(i = 0;i < 2;i++)
{
pid_t pid = fork();
}
使用上面的方法创建的子进程个数为2的n次方减1个,因为在父进程创建完子进程后子进程也会进行创建子进程。
int i = 0;
for(i = 0;i < 2;i++)
{
pid_t pid = fork();
if(pid == 0)//防止子进程创建子进程
break;
}
可以判断 i 的值来确定当前子进程为那个,用于区分子进程需要完成的任务
3.进程组
- 进程组ID为第一个进程ID(该进程则为组长进程)
- 父进程创建子进程时会默认将子进程和父进程设为同一组
- 使用kill -SIGKILL -进程组ID来杀死整个进程组中的进程
//获得当前进程的进程组ID
#include <unistd.h>
pid_t getpgrp(void);
//获取指定进程的进程组ID
pid_t getpgrp(pid_t pid);
//将进程添加到一个现有或创建的进程组中
//参数1为对应的进程,参数2为
int setpgrp(pid_t pid,pid_t pgid);
4.会话
- 会话是一个或多个进程组的集合,一个会话可以有一个控制终端。
- 当建立与控制终端连接的会话首个进程为控制进程
- 会话如果有一个控制终端,则进程组中一个为前台进程组(即有控制终端权限的),其余为后台进程组
注意:
- 会话首进程不能是进程组组长,其会成为一个新的进程组组长
- 需要root权限(ubuntu不需要)
- 控制终端只能存在一个会话中
创建会话:
先调用fork,终止父进程(使用exit(-1)来结束),子进程调用setsid,将其设为会话
#include <unistd.h>
//获取进程所属的会ID
pid_t getsid(pid_t pid);
//创建会话
pid_t setsid();
5.创建守护进程
创建步骤:
- 创建子进程,退出父进程
- 在子进程中创建新会话
- 改变当前目录为根目录(不必须)chdir函数:防止占用可卸载的文件系统
- 重设文件权限掩码(不必须)umask函数:防止进程的文件创建屏蔽字拒绝某些权限,增加守护进程灵活性
- 关闭文件描述符(不必须):防止浪费系统资源
- 开始执行守护进程核心工作
int main(int argc,char const *argv[]){
pid_t pid = fork();
//结束父进程
if(pid > 0)
exit(-1);
//设为会话
setsid();
//改变工作目录
chdir("/");
//设置权限掩码
umask(0002);
//关闭文件描述符0 1 2
close(0);
close(1);
close(2);
//守护进程的核心任务
//代码
}