Linux下的进程控制原语【pid_t,fork,exec,wait,waitpid,getpid,kill,pasue,sleep,signal】
目录
一、pid_t 进程号类型
二、与进程创建、执行有关的系统调用说明
fork()系统调用:
exec 系统调用:
wait() 和 waitpid() 系统调用:
getpid()系统调用语法:
三、与进程控制有关的系统调用说明
kill 系统调用:
pause 系统调用:
sleep 系统调用:
signal 系统调用:
四、进程示例实验
一、pid_t 进程号类型
经常碰到一个语句:pid_t pid = fork();
这里的pid_t是一个类型,就像int型一样,int型定义的变量都是整型的,pid_t定义的类型都是进程号类型。这个语句的意思是定义了一个pid_t类型的变量pid,fork()函数返回一个进程号,这个进程号赋给了pid。pid_t在头文件types.h(sys/types.h)中定义。
pid_t是一个typedef定义类型,用它来表示进程id类型。
sys/types.h:
typedef short pid_t; /* used for process ids */
pid_t就是一个short类型变量,实际表示的是内核中的进程表的索引。
使用pid_t而不使用int只是为了可移植性好一些.因为在不同的平台上有可能 typedef int pid_t,也有可能 typedef long pid_t。
二、与进程创建、执行有关的系统调用说明
进程可以通过系统调用 fork()创建子进程并和其子进程并发执行.子进程初始的执行映像是父进程的一个复本.子进程可以通过 exec()系统调用族装入一个新的执行程序。父进程可以使用 wait()或 waitpid()系统调用等待子进程的结束并负责收集和清理子进程的退出状态。
fork()系统调用:
#include <unistd.h>
pid_t fork(void);
fork 成功创建子进程后将返回子进程的进程号,不成功会返回-1.
exec 系统调用:
exec 系统调用 有一组 共6 个函数,其中示例实验中引用了 execve() 系统调用语法:
#include <unistd.h>
int execve(const char *path, const char *argv[], const char * envp[]);
- path 要装入的新的执行文件的绝对路径名字符串.
- argv[] 要传递给新执行程序的完整的命令参数列表(可以为空).
- envp[] 要传递给新执行程序的完整的环境变量参数列表(可以为空).
Exec 执行成功后将用一个新的程序代替原进程,但进程号不变,它绝不会再返回到调用进程了。如果 exec 调用失败,它会返回-1。
wait() 和 waitpid() 系统调用:
#include <sys/types.h>
#include <sys/wait.h>
pid_t wait(int *status);
pid_t waitpid(pid_t pid,int *status,int option);
status 用于保留子进程的退出状态。
pid可以为以下可能值:
- -1 等待所有 PGID 等于 PID 的绝对值的子进程
- 1 等待所有子进程
- 0 等待所有 PGID 等于调用进程的子进程
- >0 等待 PID 等于 pid 的子进程
说明:PGID(Process Group ID 进程组 ID号)
option 规定了调用 waitpid 进程的行为:
- WNOHANG 没有子进程时立即返回
- WUNTRACED 没有报告状态的进程时返回
wait() 和 waitpid() 系统调用执行成功将返回终止的子进程的进程号,不成功返回-1。
wait 和 waitpid 区别:
- 在一个子进程终止前,wait使其调用者阻塞,而waitpid有一个选项,可使调用者不阻塞;
- waitpid()并不等待在其调用之后的第一个终止的子进程,它有若干个选项,可以控制它所等待的进程;
getpid()系统调用语法:
#include <sys/types.h>
#include <unistd.h>
pid_t getpid(void);
pid_t getppid(void);
getpid 返回当前进程的进程号,getppid 返回当前进程父进程的进程号。
三、与进程控制有关的系统调用说明
可以通过信号向一个进程发送消息以控制进程的行为。信号是由中断或异常事件引发的,如:键盘中断、定时器中断、非法内存引用等。
信号的名字都以 SIG 开头,例如 SIGTERM、SIGHUP。可以使用 kill -l 命令查看系统当前的信号集合。
信号可在任何时间发生,接收信号的进程可以对接收到的信号采取3种处理措施之一:
- 忽略这个信号
- 执行系统默认的处理
- 捕捉这个信号做自定义的处理
信号从产生到被处理所经过的过程:
产 生 (generate)-> 挂 起 (pending)-> 派 送 (deliver)-> 部 署 (disposition) 或忽略(igore)
一个信号集合是一个 C 语言的 sigset_t 数据类型的对象,sigset_t 数据类型定义在<signal.h>中。被一个进程忽略的所有信号的集合称为一个信号掩码(mask)。
从程序中向一个进程发送信号有两种方法:调用 shell 的 kill 命令,调用kill系统调用函数。
kill能够发送除杀死一个进程(SIGKILL、SIGTERM、SIGQUIT)之外的其他信号,例如键盘中断(Ctrl+C)信号 SIGINT,进程暂停(Ctrl+Z)信号SIGTSTP 等等。
调用 Pause 函数会令调用进程的执行挂起直到一个任意信号到来后再继续运行。
调用 sleep 函数会令调用进程的执行挂起睡眠指定的秒数或一个它可以响应的信号到来后继续执行。
每个进程都能使用 signal 函数定义自己的信号处理函数,捕捉并自行处理接收的除 SIGSTOP 和 SIGKILL 之外的信号。
以下是有关的系统调用的语法说明。
kill 系统调用:
#include <sys/types.h>
#include <signal.h>
int kill(pid_t pid, int sig);
- pid 接收信号的进程号
- signal 要发送的信号
kill 发送成功返回接收者的进程号,失败返回-1。
pause 系统调用:
#include <unistd.h>
int pause(void);
pause 挂起调用它的进程直到有任何信号到达。调用进程不自定义处理方法,则进行信号的默认处理。
只有进程自定义了信号处理方法捕获并处理了一个信号后,pause 才会返回调用进程。
pause 总是返回-1,并设置系统变量 errno 为 EINTR。
sleep 系统调用:
#include <unistd.h>
unsigned int sleep(unsigned int seconds);
seconds 指定进程睡眠的秒数。如果指定的秒数到,sleep 返回 0。
signal 系统调用:
#include <signal.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler);
- signum 要捕捉的信号
- handler 进程中自定义的信号处理函数名
signal 调用成功会返回信号处理函数的返回值,不成功返回-1,并设置系统变量 errno 为 SIG_ERR。
四、进程示例实验
以下实验示例程序应实现一个类似子 shell 子命令的功能,它可以从执行程序中启动另一个新的子进程并执行一个新的命令和其并发执行.
1) 打开一终端命令行窗体,新建一个文件夹,在该文件夹中建立以下名为pctl.c的C语言程序:
#include "pctl.h"int main(int argc, char *argv[]) { int i;int pid; //存放子进程号int status; //存放子进程返回状态char *args[] = {"/bin/ls","-a",NULL}; //子进程要缺省执行的命令signal(SIGINT,(sighandler_t)sigcat); //注册一个本进程处理键盘中断的函数pid=fork() ; //建立子进程if(pid<0){ // 建立子进程失败printf("Create Process fail!\n");exit(EXIT_FAILURE);}if(pid == 0){// 子进程执行代码段printf("I am Child process %d\nMy father is %d\n",getpid(),getppid()); //报告父子进程进程号pause(); //暂停,等待键盘中断信号唤醒printf("%d child will Running: \n",getpid()); //子进程被键盘中断信号唤醒继续执行if(argv[1] != NULL){//如果在命令行上输入了子进程要执行的命令,则执行输入的命令for(i=1; argv[i] != NULL; i++) printf("%s ",argv[i]); printf("\n"); //装入并执行新的程序status = execve(argv[1],&argv[1],NULL);} else {//如果在命令行上没输入子进程要执行的命令 则执行缺省的命令for(i=0; args[i] != NULL; i++) printf("%s ",args[i]); printf("\n"); //装入并执行新的程序status = execve(args[0],args,NULL);}} else {//父进程执行代码段printf("\nI am Parent process %d\n",getpid()); //报告父进程进程号if(argv[1] != NULL){ //如果在命令行上输入了子进程要执行的命令 则父进程等待子进程执行结束printf("%d Waiting for child done.\n\n",getpid()); waitpid(pid,&status,0); //等待子进程结束printf("\nMy child exit! status = %d\n\n",status); }else{ //如果在命令行上没输入子进程要执行的命令 唤醒子进程,与子进程并发执行不等待子进程执行结束if(kill(pid,SIGINT) >= 0) printf("%d Wakeup %d child.\n",getpid(),pid) ;printf("%d don't Wait for child done.\n\n",getpid()); }}return EXIT_SUCCESS;
}
2) 再建立以下名为 pctl.h 的 C 语言头文件:
#include <sys/types.h>
#include <wait.h>
#include <unistd.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
//进程自定义的键盘中断信号处理函数
typedef void (*sighandler_t) (int);
void sigcat(){printf("%d Process continue\n",getpid());
}
head = pctl.h
srcs = pctl.c
objs = pctl.o
opts = -g -c
all: pctl
pctl: $(objs)gcc $(objs) -o pctl
pctl.o: $(srcs) $(head)gcc $(opts) $(srcs)
clean:rm pctl *.o
4) 输入 make 命令编译连接生成可执行的 pctl 程序
$ gmake
gcc -g -c pctl.c
gcc pctl.o -o pctl
5) 执行 pctl 程序(注意进程号是动态产生的,每次执行都不相同)
I am Child process 4113My father is 4112I am Parent process 4112第二部分 操作系统算法实验Wakeup 4113 child.4112 don't Wait for child done.4113 Process continue4113 child will Running: /bin/ls -a
以上程序的输出说明父进程 4112 创建了一个子进程 4113,子进程执行被暂停。父进程向子进程发出键盘中断信号唤醒子进程并与子进程并发执行。父进程并没有等待子进程的结束继续执行先行结束了(此时的子进程成为了孤儿进程,不会有父进程为它清理退出状态了)。而子进程继续执行,它变成了列出当前目录所有文件名的命令 ls -a。在完成了列出文件名命令之后,子进程的执行也结束了。此时子进程的退出状态将有初始化进程为它清理。
6) 再次执行带有子进程指定执行命令的 pctl 程序:
$ ./pctl /bin/ls -l
I am Child process 4223
My father is 4222
I am Parent process 4222
4222 Waiting for child done.
可以看到这一次子进程仍然被挂起,而父进程则在等待子进程的完成。为了检测父子进程是否都在并发执行,请输入 ctrl+z 将当前进程放入后台并输入 ps 命令查看当前系统进程信息,显示如下:
[1]+ Stopped ./pctl /bin/ls -l $ ps -l
F S UID PID PPID C PRI NI ADDR SZ WCHAN TTY TIME CMD
0 S 0 4085 4083 0 76 0 - 1413 wait pts/1 00:00:00 bash
0 T 0 4222 4085 0 76 0 - 360 finish pts/1 00:00:00 pctl
1 T 0 4223 4222 0 76 0 - 360 finish pts/1 00:00:00 pctl
0 R 0 4231 4085 0 78 0 - 1302 - pts/1 00:00:00 ps
可以看到当前系统中同时有两个叫 pctl 的进程,它们的进程号分别是 4222和 4223。它们的状态都为―T‖,说明当前都被挂起。4223 的父进程是 4222, 而 4222 的父进程是 4085,也就是 bash-shell。为了让 pctl 父子进程继续执行,请输入 fg 命令让 pctl 再次返回前台,显示如下:$ fg
./pctl /bin/ls -l
现在pctl父子进程从新返回前台。我们可以通过键盘发键盘中断信号来唤醒pctl 父子进程继续执行,输入 ctrl+c,将会显示:
4222 Process continue
4223 Process continue
4223 child will Running: /bin/ls -l
total 1708
-rw-r--r-- 1 root root 176 May 8 11:11 Makefile
-rwxr-xr-x 1 root root 8095 May 8 14:08 pctl
-rw-r--r-- 1 root root 2171 May 8 14:08 pctl.c
-rw-r--r-- 1 root root 269 May 8 11:10 pctl.h
-rw-r--r-- 1 root root 4156 May 8 14:08 pctl.o
My child exit! status = 0
以上输出说明了子进程在捕捉到键盘中断信号后继续执行了指定的命令,按我们要求的长格式列出了当前目录中的文件名,父进程在接收到子进程执行结束的信号后将清理子进程的退出状态并继续执行,它报告了子进程的退出编码(0 表示子进程正常结束)最后父进程也结束执行。
Linux下的进程控制原语【pid_t,fork,exec,wait,waitpid,getpid,kill,pasue,sleep,signal】相关推荐
- Linux下 prctl 进程控制
Linux 下 prctl 进程控制 Linux下可通过 prctl进行进程各种控制 #include <sys/prctl.h>int prctl(int option, unsigne ...
- 操作系统实验:Linux下的进程控制实验
进程控制实验 一.实验目的: 二.实验平台: 三.实验内容: 1.进程的创建与销毁 进程控制相关函数 实验结果分析 2.多进程并发执行 time命令 实验结果分析 四.总结分析 一.实验目的: 加深对 ...
- 【Linux系统编程学习】Linux进程控制原语(fork、exec函数族、wait)
此为牛客Linux C++和黑马Linux系统编程课程笔记. 1. fork函数 1.1 fork创建单个子进程 #include<unistd.h> pid_t fork(void); ...
- 【Linux】从冯诺依曼体系到初识Linux下的进程
目录 前言 1.冯诺依曼体系结构 2.管理和操作系统 3.初识进程 1.描述进程PCB(process control block) 1.标识符pid fork接口创建子进程 2.进程状态 3.进程优 ...
- Supervisor-类unix系统下的进程控制工具
如果你的英文足够好,请看官网的文档:http://supervisord.org/introduction.html 简介: Supervisor 类unix系统下的进程控制工具. 特性: 1.配置简 ...
- linux系统调用劫持隐藏进程,linux 下隐藏进程的一种方法及遇到的坑
前言 1.本文所用到的工具在 https://github.com/gianlucaborello/libprocesshider 可以下载 2.思路就是利用 LD_PRELOAD 来实现系统函数的劫 ...
- linux下僵尸进程(<defunct>进程)的产生与避免
一.什么是僵尸进程 在UNIX 系统中,一个进程结束了,但是他的父进程没有等待(调用wait / waitpid)他,那么他将变成一个僵尸进程.当用ps命令观察进程的执行状态时,看到这些进程的状态栏为 ...
- Linux第二次试验:Linux下的进程通信实验
Linux第二次试验:Linux下的进程通信实验 前言 一.实验目的 二.实验工具与设备 三.实验预备知识 三.实验内容和步骤 五.实验代码及步骤截图 六.实验总结 前言 为了帮助同学们完成痛苦的实验 ...
- linux 下得到进程的启动时间
linux 下得到进程的启动时间! 运行方式:./pstart 进程号 " 如: ./pstart 1 #!/bin/bash pid=$1 if [ "$pid" == ...
最新文章
- Oracle常用工具
- Window10+VS2015+DevExpress.net 15.1.7完美破解(图)
- 带卷积核二分类网络的输出是不是有方向的?
- win7下安装rose
- js 更改json的 key
- [转载] Python numpy函数:all()和any()比较矩阵
- python大神的成长之路_我的Python成长之路
- 22. PE结构-PE详解之输入表(导入表)、屠龙刀W32Dasm(静态)、LordPE(动态)工具入门(查找dll、调用函数)
- Python基础——正则表达式
- 编译HG255D的openwrt固件
- 谷歌浏览器、Yandex浏览器使用体验分享
- 专利查询下载的几个网站
- 新点互联互通_新点驱动(江苏省互联互通版)
- android 获取设备的mac地址,Android编程获取设备MAC地址的实现方法
- matplotlib之pyplot模块之坐标轴配置(axis():设置坐标轴外观、设置坐标轴范围)
- 【面试 - 八股文】Linux 高频面试题,助你吊打面试官系列
- java 第三方登录之QQ登录
- 如何配置在线Yum源?
- 某大厂来的水货CTO,写出了新手都不会犯的低级 Bug,被骗300W!就里就不点名了!...
- 接到面试通知后的准备
热门文章
- 2023上海锂电池技术展览会|新能源动力电池及储能电池博览会|长三角锂电池材料设备展会
- jQuery之标签操作和返回顶部、登录验证、全选反选、克隆示例
- ClickHouse表引擎详解
- 实变函数论知识点总结
- LeetCode常见题型——二分查找
- 【北京理工大学-Python 数据分析-2.1Matplotlib库入门】
- 原版 openwrt + WireGuard + passwall 编译
- Java爬虫抓取豆瓣读书信息
- c# 实现一个在线电台的功能 急求.....
- 做一个物联网温湿度传感器(一)SHT30传感器介绍