APUE(8)---进程控制(1)
一、进程标识
每个进程都有一个非负整型标识的唯一进程ID。因为进程ID标识符总是唯一的,常将其用做其他标识符的一部分以保证其唯一性。进程ID虽然是唯一的, 但是却是可以复用的。ID为0的进程通常是调度进程,常常被称为交换进程(swapper)。该进程是内核的一部分,它并不执行任何磁盘上的程序,因此也被称为系统进程。进程ID为1通常是init进程,在自举过程结束时由内核调用。此进程负责在自举内核后启动一个UNIX系统,init通常读取与系统有关的初始化文件,并将系统引导一个状态。init进程绝不会终止。它是一个普通的用户进程,但是它以超级用户特权运行。进程ID为2是页守护进程,此进程负责支持虚拟存储器系统的分页操作。
#include <unistd.h>
pid_t getpid(void); //返回进程的进程ID
pid_t getppid(void); //返回进程的父进程ID
pid_t getuid(void); //返回进程的实际用户ID
pid_t geteuid(void); //返回调用进程的有效用户ID
pid_t getgid(void); //返回进程的实际组ID
pid_t getepid(void); //返回进程的有效组ID
二、函数fork
#include <unistd.h>
pid_t fork(void);
由fork创建的新进程被称为子进程(child process)。fork函数被调用一次,但返回两次。两次返回的区别是子进程的返回值是0,而父进程的返回值则是新建子进程的进程ID。原因:因为一个进程的子进程可以有很多个,并且没有一个函数使一个进程可以获得其所有子进程的进程ID;一个进程只会有一个父进程,所以子进程总是可以调用getppid以获得其父进程的进程ID。子进程是父进程的副本,子进程获得父进程数据空间、堆和栈的副本,父进程和子进程不共享这些存储空间部分。父进程和子进程只共享正文段。由于在fork之后经常跟随着exec,所以现在的很多实现并不执行一个父进程数据段和栈和堆的完全副本,作为替代,使用了写时复制(Copy-On-Write)技术。
#include "apue.h"
int globvar = ;
char buff[] = "a write to stdout\n"; int main(void)
{
int var;
pid_t pid; var = ;
if(write(STDOUT_FILENO, buf, sizeof(buf) - ) != sizeof(buf) - )
{
err_sys("write error");
} printf("before fork\n"); if((pid = fork()) < )
{
err_sys(" fork error");
}
else if(pid == )
{
globvar ++;
var ++;
}
else
{
sleep();
} printf("pid = %ld, glob = %d, var = %d\n", (long)getpid(), globvar, var);
exit();
}
8-1:fork函数实例
fork的一个特性是父进程的所有打开文件描述符都被复制到子进程中。我们说赋值是因为对每个文件描述符来说,就好像执行了dup函数,父进程和子进程每个相同的打开描述符共享一个文件表项。重要的一点是:父进程和子进程共享同一个文件偏移量。
8-2: fork之后父进程和子进程之间对打开文件的共享
除了打开文件之外,父进程的很多其他属性也由子进程继承:实际用户ID、实际组ID、有效用户ID、有效组ID;附属组ID;进程组ID;会话ID;控制终端;设置用户ID标志和设置组ID标志;当前工作目录;根目录;文件模式创建屏蔽字;信号屏蔽和安排;对任一打开文件描述符的执行时关闭标志;环境;连接的共享存储段;存储映像;资源限制。父进程和子进程之间的区别在于:fork的返回值不同;进程ID不同;两个进程的父进程ID不同;自己称tms_utime、tms_stime、tms_ustime的值设置为0;子进程不继承父进程设置的文件所;子进程的未处理闹钟被清除;子进程的未处理信号集设置为空集。
fork失败的两个主要原因:系统中已经有了太多的进程;该实际用户ID的进程总数超过了系统限制。fork有以下两个用法:一个父进程希望复制自己,使父进程和子进程执行不同的代码段;一个进程要执行一个不同的程序。
三、函数vfork
vfork与fork一样都创建一个子进程,但是它并不将父进程的地址空间完全复制到子进程中,因为子进程会立即调用exec,预示也就不会引用该地址空间;vfork和fork之间的另一个区别是:vfork保证子进程先运行,在它调用exec或exit之后父进程才可能被调度运行,当子进程调用这两个函数中的任意一个时,父进程会恢复运行。
#include "apue.h" int globvar = ;
int main(void)
{
int var;
pid_t pid; var = ;
pirntf("before vfork\n");
if((pid = vfork()) < )
{
err_sys("vfork error");
}
else if(pid == )
{
globvar ++;
var ++;
_exit();
} pirntf("pid = %ld, glob = %d, var = %d\n",(long)getpid(), globvar, var);
exit();
}
8-3: vfork函数实例
四、函数exit
在大多数UNIX系统实现中,exit是标准C库中的一个函数,而_exit则是一个系统调用。不管进程如何终止,最后都会执行内核中的同一段代码。这段代码为相应进程关闭所有打开描述符,释放它所使用的存储器等。在任意一种情况下,该终止进程的父进程都能用wait或waipid函数取得其终止状态。对于父进程已经终止的所有进程,它们的父进程都改变为init进程。我们称这些进程由init进程手痒。在UNIX术语中,一个已经终止、但是父进程尚未对其进行善后处理(获取终止子进程的有关信息、释放它仍占用的资源)的进程被称为僵死进程。
五、函数wait和waitpid
当一个进程正常或异常终止时,内核就向父进程发送SIGCHLD信息。因为子进程终止是个异步事件,所以这种信号也是内核向父进程发的异步通知。父进程可以选择忽略该信号,或者提供一个该信号发生时即被调用执行的函数,对这种信号的系统默认动作是忽略它。
#include <sys/wait.h>
pid_t wait(int *statloc);
pid_t waitpidd(pid_t pid, int *statloc, int options);
如果其所有子进程都还在运行,则阻塞;如果一个子进程已终止,正等待父进程获取其终止状态,则取得该子进程的终止状态立即返回;如果它没有任何子进程,则立即出错返回。
#include "apue.h"
#include <sys/wait.h> void pr_exit(int status); int main(void)
{
pid_t pid;
int status; if((pid = fork) < )
{
err_sys("fork error");
}
else if(pid == )
{
exit();
} if(wait(&status) != pid)
{
err_sys("wait error");
}
pr_exit(status); if((pid = fork) < )
{
err_sys("fork error");
}
else if(pid == )
{
abort();
} if(wait(&status) != pid)
{
err_sys("wait error");
}
pr_exit(status); if((pid = fork) < )
{
err_sys("fork error");
}
else if(pid == )
{
status /= ;
} if(wait(&status) != pid)
{
err_sys("wait error");
}
pr_exit(status); exit();
} void pr_exit(int status)
{
if(WIFEXITED(status))
{
printf("normal termination, exit status = %d\n", WEXITSTATUS(status));
}
else if(WIFSIGNALED(status))
{
printf("abnormal termination, signal number = %d\n", WTERMSIG(status),
#ifdef WCORDUMP
WCORDUMP(status)? "core file generated":"");
#else
"");
#endif
}
else if(WIFSTOPPED(status))
{
printf("child stopped, signal number = &d\n", WSTOPSIG(status));
}
}
8-6:演示不同的exit值
如果一个进程有几个子进程,那么只要有一个子进程终止,wait就返回。waitpid函数提供了wait函数没有提供的3个功能:1、waitpid可等待一个特定的进程,而wait则返回任一终止子进程的状态;2、waitpid提供了一个wait的非阻塞版本;3、waitpid通过WUNTRACED和WCONTINUED选项支持作业控制。
#include "apue.h"
#include <sys/wait.h> int main(void)
{
pid_t pid;
if((pid = fork()) < )
{
err_sys("fork error");
}
else if(pid == )
{
if((pid = fork()) < )
{
err_sys("fork error");
}
else if(pid > )
{
exit();
} sleep();
printf("second child, parent pid = %ld\n",(long) getppid());
exit();
} if(waitpid(pid, NULL, ) != pid)
{
err_sys("waitpid error");
} exit();
}
8-8:fork两次以避免僵死进程
六、函数exec
当进程调用一种exec函数时,该进程执行的程序完全替换为新程序,而新程序则从其main函数开始执行。因为调用exec并不创建新进程,所以前后的进程ID并未改变。exec只是用磁盘上的一个新程序替换了当前进程的正文段、数据段、堆段和栈段。用fork可以创建新进程,用exec可以初始化执行新的程序。exit函数和wait函数处理终止和等待终止。这些是我们需要的基本的进程控制原语。
#include <unistd.h>
extern char **environ;
int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ..., char * const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *file, char *const argv[], char *const envp[]);

8-15:7个exec函数之间的关系
#include "apue.h"
#include <sys/wait.h> char *env_init[] = {"USER=unknow", "PATH=/tmp", NULL}; int main(void)
{
pid_t pid;
if((pid = fork()) < )
{
err_sys("fork error");
}
else if(pid == )
{
if(execle("/home/sar/bin/echoall", "echoall", "myarg1", "MY ARG2", (char *), env_init) < )
{
err_sys("execle error");
}
} if(waitpid(pid, NULL, ) < )
{
err_sys("waitpid error");
} if((pid == fork()) < )
{
err_sys("fork error");
}
else if(pid == )
{
if(execlp("echoall", "echoall", "only 1arg", (char *)) < )
{
err_sys("execle error");
}
} exit();
}
8-16:exec函数实例
APUE(8)---进程控制(1)的更多相关文章
- (六) 一起学 Unix 环境高级编程 (APUE) 之 进程控制
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- [08]APUE:进程控制
[a] getpid / getppid / getuid / geteuid / getgid / getegid #include <unistd.h> pid_t getpid(vo ...
- (五) 一起学 Unix 环境高级编程 (APUE) 之 进程环境
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (七) 一起学 Unix 环境高级编程(APUE) 之 进程关系 和 守护进程
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- (十) 一起学 Unix 环境高级编程 (APUE) 之 线程控制
. . . . . 目录 (一) 一起学 Unix 环境高级编程 (APUE) 之 标准IO (二) 一起学 Unix 环境高级编程 (APUE) 之 文件 IO (三) 一起学 Unix 环境高级编 ...
- 进程控制(Note for apue and csapp)
1. Introduction We now turn to the process control provided by the UNIX System. This includes the cr ...
- apue学习笔记(第八章 进程控制)
本章介绍UNIX系统的进程控制,包括创建新进程.执行程序和进程终止. 进程标识 每一个进程都有一个非负整数表示的唯一进程ID,除了进程ID,每个进程还有一些其他标识符.下列函数返回这些标识符 #inc ...
- 《UNIX环境高级编程》(APUE) 笔记第八章 - 进程控制
8 - 进程控制 Github 地址 1. 进程标识 每个进程都有一个非负整型表示的 唯一进程 ID .进程 ID 是可复用的(延迟复用算法). ID 为 \(0\) 的进程通常是调度进程,常常被称为 ...
- 进程控制之exec函数
用fork函数创建子进程后,子进程往往要调用一种exec函数以执行另一个程序.当进程调用一种exec函数时,该进程执行的程序完全替换为新程序,而新程序则从其main函数开始执行.因为调用exec并不创 ...
随机推荐
- Oracle查看和修改连接数
1.查询数据库当前进程的连接数: select count(*) from v$process; 2.查看数据库当前会话的连接数: elect count(*) from v$sessio ...
- IDA Pro 权威指南学习笔记(十二) - IDA 中的注释
注释有助于以一种更高级的方式描述汇编语言指令序列 IDA 提供了几种不同类型的注释,每种注释适用于不同的目的 使用 Edit -> Comments 命令的选项,可以为反汇编代码清单中的任何一行 ...
- [BZOJ][CQOI2014]数三角形
Description 给定一个nxm的网格,请计算三点都在格点上的三角形共有多少个.下图为4x4的网格上的一个三角形. 注意三角形的三点不能共线. Input 输入一行,包含两个空格分隔的正整数m和 ...
- 抛java.lang.NoClassDefFoundError: org.joda.time.ReadablePeriod错误
转自:http://www.codeorg.cn/article/detail/qa/542 在进行activiti环境搭建时总是抛出java.lang.NoClassDefFoundError: o ...
- Weak References
http://docwiki.embarcadero.com/RADStudio/Seattle/en/Automatic_Reference_Counting_in_Delphi_Mobile_Co ...
- RocketMQ初探(四)之RocketMQ4.x版本可视化管理控制台rocketmq-console-ng搭建(Apache)
之前有部署过3.2.6为AliBaba版本的Web监控平台(可参考之前博客 https://www.cnblogs.com/buyige/p/9395453.html),现用RocketMQ4.2.0 ...
- then()方法是异步执行
then()方法是异步执行 就是当.then()前的方法执行完后再执行then()内部的程序 这样就避免了,数据没获取到等的问题
- 修改SecureCRT终端的Home和End功能键。
SecureCRT真是个不错的ssh客户端工具,但在使用时发现跟自己的一些使用习惯不符合,例如home.end.pageup.pagedown和delete等键. 默认情况下一些按键的功能如下: pa ...
- mybatis与springdata的一些简单比较与思考
主题 最近在用mybatis做项目,有一些感触想记录下,主要是mybatis(以及它的一些插件)相比较于Spring data(或者jpa,hibernate等)的优势地方. 感触 我觉得mybati ...
- git 撤销 merging
当我们在合代码的时候经常会遇到一些问题,这时候分支就处于merging状态,这时候可以用下面的命令撤销 $ git reset --hard HEAD (or sha_1) 不知道有没有更好的办法,希 ...