UNIX进程组,会话期,作业控制(转…

2023-10-14 02:08

本文主要是介绍UNIX进程组,会话期,作业控制(转…,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

转自:http://202.117.3.13/wordpress/?p=102

在UNIX系统中,作业控制允许在一个终端上启动多个作业(进程组),控制哪一个作业可以存取该终端,以及哪些作业在后台运行。
为了支持作业控制,引入了进程组,会话期,控制终端等概念,还需要内核以一定的信号支持。

一·进程组
每一个进程除了有一个进程PID之外,还属于一个进程组,用进程组ID表示。返回当前进程组ID的系统调用为:

pid_t getpgrp();

每个进程组都有一个组长进程,组长进程的标识是进程组ID等于其进程ID
进程可以调用setpgid系统调用参加一个现存的组或者创建一个新的进程组。

int setpgid(pid_t pid, pid_t pgid);

这将pid进程的进程组ID设置为pgid,如果两者相等,则pid变为进程组的组长。
一个进程只能为它自己或者它的子进程改变进程组ID,如果pid为0,则代表自己,如果pgid为0,则由pid指定的进程ID作为进程组ID。
如果pid和pgid不等,而目前系统中不存在pgid的进程组,则出错。
当用fork()产生一个子进程后,子进程将继承父进程的进程组ID,也就是子进程和父进程属于同一个进程组。

二·对话期(session)
对话期是一个或多个进程组的集合,对话期可以有一个控制终端。例如,可以由以下的安排:
UNIX进程组,会话期,作业控制(转载)

进程调用setsid函数可以创建一个新的对话期。

pid_t setsid();

如果调用此函数的进程是一个进程组的组长,则出错。否则该函数创建一个新的对话期,结果为:
1)该进程变为新的对话期的首进程。
2)此进程成为一个新进程组的组长进程。新进程组的ID为调用进程的进程ID。
3)此进程没有控制终端。

三。前台进程组,后台进程组
一个对话期的几个进程组可以被分成一个前台进程组以及一个或几个后台进程组。
如果一个对话期有一个控制终端,那么它有一个前台进程组,其他进程组为后台进程组。
无论何时键入中断键(Ctrl-C)或者退出键(Ctrl-\),就会造成中断信号SIGINT或者退出信号SITQUIT送至前台进程组中的所有进程。
只有前台进程组中的进程可以接受终端输入,如果后台进程组的进程试图读终端,那么内核会发送一个特定的信号SIGTTIN给后台作业,这通常会停止(挂起)次后台作业。当用将次后台进程转为前台进程后(移入前台进程组),会发送一个SIGCONT信号给该进程,使该进程继续运行。

四·测试

#include
#include
#include
#include
#include
 
int main() {
setbuf(stdout, NULL);
 
printf("main: %d %d\n", getpid(), getpgrp());
 
pid_t pid = fork();
if(pid < 0) {
perror("fork");
return 1;
}
 
setpgid(pid, 0);
if(pid > 0) { //pid > 0则进入父进程处理分支,pid 是子进程的进程标识符。
sleep(5);
setpgid(pid, getpgrp());
kill(pid, SIGCONT);
waitpid(pid, NULL, 0);
return 0;
}
 
char buf[1024];
printf("child: %d %d\n", getpid(), getpgrp());
while(fgets(buf, 1024, stdin)) {
fputs(buf, stdout);
}
return 0;
}

该程序首先打印父进程的PID进程组ID,由于该进程由shell创建,所以会将该进程的进程组ID设置为进程PID,是的该进程组属于一个新进程组,并且为前台进程组。然后fork出一个子进程,此时子进程应该继承父进程的进程组ID,和父进程同属于前台进程组,然后父子进程下一步都调用 setpgid(pid, 0),这会确保把子进程设为一个新的进程组的组长,并且该进程组为后台进程组。这个时候子进程打印自己的进程PID和进程组ID,然后开始循环从终端读入一行数据,并原样输入到终端,由于这个子进程属于后台进程组,这会导致子进程被挂起(停止),所以屏幕上不会有什么输出。父进程先睡眠5秒钟(给我足够的时间来敲几行字符演示子进程确实没有输出),然后设置子进程的组ID为自己(父进程)的组ID,也就是将子进程移入前台进程组,然后发送SIGCONT信号给子进程,使子进程重新运行,由于这是子进程已经属于前台进程组了,因此可以成功的读入终端字符并显示出来。

运行结果:
UNIX进程组,会话期,作业控制(转载)

这篇关于UNIX进程组,会话期,作业控制(转…的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/207392

相关文章

[Linux]:进程(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 进程终止 1.1 进程退出的场景 进程退出只有以下三种情况: 代码运行完毕,结果正确。代码运行完毕,结果不正确。代码异常终止(进程崩溃)。 1.2 进程退出码 在编程中,我们通常认为main函数是代码的入口,但实际上它只是用户级

java 进程 返回值

实现 Callable 接口 与 Runnable 相比,Callable 可以有返回值,返回值通过 FutureTask 进行封装。 public class MyCallable implements Callable<Integer> {public Integer call() {return 123;}} public static void main(String[] args

C#关闭指定时间段的Excel进程的方法

private DateTime beforeTime;            //Excel启动之前时间          private DateTime afterTime;               //Excel启动之后时间          //举例          beforeTime = DateTime.Now;          Excel.Applicat

linux中使用rust语言在不同进程之间通信

第一种:使用mmap映射相同文件 fn main() {let pid = std::process::id();println!(

列举你能想到的UNIX信号,并说明信号用途

信号是一种软中断,是一种处理异步事件的方法。一般来说,操作系统都支持许多信号。尤其是UNIX,比较重要应用程序一般都会处理信号。 UNIX定义了许多信号,比如SIGINT表示中断字符信号,也就是Ctrl+C的信号,SIGBUS表示硬件故障的信号;SIGCHLD表示子进程状态改变信号;SIGKILL表示终止程序运行的信号,等等。信号量编程是UNIX下非常重要的一种技术。 Unix信号量也可以

Golang进程权限调度包runtime

关于 runtime 包几个方法: Gosched:让当前线程让出 cpu 以让其它线程运行,它不会挂起当前线程,因此当前线程未来会继续执行GOMAXPROCS:设置最大的可同时使用的 CPU 核数Goexit:退出当前 goroutine(但是defer语句会照常执行)NumGoroutine:返回正在执行和排队的任务总数GOOS:目标操作系统NumCPU:返回当前系统的 CPU 核数量 p

如何保证android程序进程不到万不得已的情况下,不会被结束

最近,做一个调用系统自带相机的那么一个功能,遇到的坑,在此记录一下。 设备:红米note4 问题起因 因为自定义的相机,很难满足客户的所有需要,比如:自拍杆的支持,优化方面等等。这些方面自定义的相机都不比系统自带的好,因为有些系统都是商家定制的,难免会出现一个奇葩的问题。比如:你在这款手机上运行,无任何问题,然而你换一款手机后,问题就出现了。 比如:小米的红米系列,你启用系统自带拍照功能后

flume系列之:记录一次flume agent进程被异常oom kill -9的原因定位

flume系列之:记录一次flume agent进程被异常oom kill -9的原因定位 一、背景二、定位问题三、解决方法 一、背景 flume系列之:定位flume没有关闭某个时间点生成的tmp文件的原因,并制定解决方案在博主上面这篇文章的基础上,在机器内存、cpu资源、flume agent资源都足够的情况下,flume agent又出现了tmp文件无法关闭的情况 二、

C++编程:ZeroMQ进程间(订阅-发布)通信配置优化

文章目录 0. 概述1. 发布者同步发送(pub)与订阅者异步接收(sub)示例代码可能的副作用: 2. 适度增加缓存和队列示例代码副作用: 3. 动态的IPC通道管理示例代码副作用: 4. 接收消息的超时设置示例代码副作用: 5. 增加I/O线程数量示例代码副作用: 6. 异步消息发送(使用`dontwait`标志)示例代码副作用: 7. 其他可以考虑的优化项7.1 立即发送(ZMQ_IM

[轻笔记] ubuntu Shell脚本实现监视指定进程的运行状态,并能在程序崩溃后重启动该程序

根据网上博客实现,发现只能监测进程离线,然后对其进行重启;然而,脚本无法打印程序正常状态的信息。自己通过不断修改测试,发现问题主要在重启程序的命令上(需要让重启的程序在后台运行,不然会影响监视脚本进程,使其无法正常工作)。具体程序如下: #!/bin/bashwhile [ 1 ] ; dosleep 3if [ $(ps -ef|grep exe_name|grep -v grep|