【APUE】Chapter7 Process Environment

这一章内容是Process的基础准备篇章。这一章的内容都是基于C Programm为例子。

(一)进程开始:

  kernel → C start-up rountine → main function

  其中C start-up rountine做的事情中包括:获得command-line arguments & environment

  好像还有其他的事情,后面再说。

(二)进程结束:

  五种normal termination方式:

    1. return from main

    2. exit

    3. _exit _Exit

    4. 从start routine调用的last pthread里面return

    5. 在最后一个pthread中调用pthread_exit

  三种abnormal termination方式:(这三种方式先不学)

    1. abort

    2. receipt of a signal

    3. response of the last pthread to a cancellation request

  注册进程退出时执行“钩子函数”:atexit

    基于stack结构实现的,所以用atexit注册的函数在进程exit的时候是reverse order执行的(跟Chapter11提到的进程清理函数也类似)

    这部分的代码比较直观,看书上P202 Figure7.3代码即可。

 APUE 英文3版201页Figure7.2是个比较综合的图,很好阐述了进程开始和结束的各种路径以及做的事情

    简单来说:

    1. return或exit :先调用各种exit handler以及standard I/O cleanup;再调用_exit或_Exit

    2. 如果是直接_exit或_Exit就直接强行退出进程,一步到位,不会调用各种exit handler和Cleanup等了

(三)Command-Line Arguments

  没有什么特别解释的 就是main传入两个参数:

  1. int argc:参数个数

  2. char *argv[]:以字符串形式传入的参数

  argv[0]永远是命令本身,从argv[1]开始是真正传入程序的参数

(四)Environment List

  之前main的参数中是有这个的;但是后来发现这东西的用途太多了,地位就升级了,就给他专门搞了一些内置的函数去操作

(五)Memory Layout of a C program

  重点都在P206的Figure7.6上面。

  地址由高到低依次是:

  第一部分:Command-Line arguments & environment variables

  第二部分:stack(栈)(调用函数的stack frame在这里,函数返回之后,这部分资源就可能瞬间被其他内容覆盖了)

  第三部分:留白 (stack可以向下grow,占用这块留白;heap可以向上grow,占用这块留白)

  第四部分:heap(堆)(提供动态分配的空间:malloc;如果忘了free,就memory leak了)

  第五部分:uninitialized data (没有初始化的全局变量)

  第六部分:initialized data (初始化的全局变量)

  第七部分:text segment(包括CPU可执行的machine instructions)

  其中,只有六、七两部分是存在于program file中的;五貌似也像在program file中的,其实不是的,是由kernel设置为0的。

(六)Shared Libraries、(七)Memory Allocation留以后用到再看

(八)Environment Variables

  由于Environment Variables的地位提高了,因此有三个函数可以改环境变量:

    1. putenv

    2. setenv

    3. unsetenv

  这里有两点需要注意下:

    1. 就是这个简单的Environment Variables,为什么要搞这样几个复杂的函数去设置、修改?回想Memory Layout of a C program,最上面那块大小是定死的:如果EV比原来短还好,比原来长,或者动态增加,就需要利用heap空间了。这就涉及到一些麻烦事情。详情书上写的挺好。

    2. 进程中设定EV的作用域:在当前进程设置的EV,只对当前进程以及其子进程有效,不影响其他进程。

(九)setjmp和longjmp

  代码都在书上,很直观就不粘上来了。

  Goto是在同一个函数中跳转(某个stack frame内);setjmp和longjmp负责在一个进程的函数间跳(夸若干个stack frame)

  有几点需要注意:

    1. setjmp设定跳回到哪,longjmp负责跳,用jmp_buf类型的变量来协调二者,跳到哪。

    2. 通一个setjmp可以从不同地方longjmp回来

    3. f1() { setjmp; f2(){f3(longjmp;)}} 如果从f3中跳回f1,则stack中f2和f3的stack frame都无效了(即不归f2和f3是换了,可能迅速被其他占用了)

    4. 还是3中的例子,如果longjmp回到f1中了,那么f1中变量的状态呢?受到执行完f1和f2之后的影响,还是保存之前的状态呢?答案是不一定,尤其是autmatic register变量,更是不一定了。这里比较好的办法是,把变量设成volatile的,这样即使longjmp回来,变量肯定是受到f2和f3的影响后的值,不会出现不确定状态

(十)getrlimit & setrlimit

  涉及到进程占用资源的分配,后面再看。

  

  

时间: 2024-08-06 02:51:01

【APUE】Chapter7 Process Environment的相关文章

【APUE】Chapter8 Process Control

这章的内容比较多.按照小节序号来组织笔记的结构:再结合函数的示例带代码标注出来需要注意的地方. 下面的内容只是个人看书时思考内容的总结,并不能代替看书(毕竟APUE是一本大多数人公认的UNIX圣经). 8.2 Process Identifiers 1. unix system给系统分配进程pid采用的是delay reuse策略:即,刚用完被释放的pid不会马上分配给新的进程,目的是防止新进程错误使用与之前进程相同的ID(这块内容还没太懂,以后再看):但具体等多久不一定. 2. 有几个特殊的p

【APUE】Chapter9 Process Relationships

这一章看的比较混乱,可能是因为例子少:再有就是,这一章就是一个铺垫的章节. 9.2 terminal logins 啥叫termnial? 我感觉书上的terminal指的更可能是一些物理设备(keyboard, modem这类的) /etc/ttys里面存着这些终端,一行代表一个终端的信息. 限于基础知识有限,上面的内容可能理解的有误,但是对于terminal logins有一点是可以确定的:计算机事先知道有多少terminals可以logins. 其中BSD Terminal Logins是

【APUE】孤儿进程与僵死进程

基本概念: 在unix/linux中,正常情况下,子进程是通过父进程创建的,子进程在创建新的进程.子进程的结束和父进程的运行是一个异步过程,即父进程永远无法预测子进程 到底什么时候结束. 当一个 进程完成它的工作终止之后,它的父进程需要调用wait()或者waitpid()系统调用取得子进程的终止状态. 孤儿进程:一个父进程退出,而它的一个或多个子进程还在运行,那么那些子进程将成为孤儿进程.孤儿进程将被init进程(进程号为1)所收养,并由init进程对它们完成状态收集工作. 僵尸进程:一个进程

【APUE】进程基础

进程标识符:非负整数 ID为0的进程通常是是调度进程,常被称为交换进程.该进程是内核的一部分,它并不执行任何磁盘上的程序,因此也被称为系统进程 ID为1的进程是init进程,在自举过程结束时由内核调用,此进程负责在自举内核后启动一个UNIX系统,init通常读入与系统有关的初始化文件,并将系统引导到一个状态,init进程不会终止.它是普通的用户进程,以超级用户特权运行 常用函数: pid_t getpid(void)  返回调用进程的进程id pid_t getppid(void) 返回调用进程

【APUE】fork函数

#include <unisth.h> pid_t fork(void) fork函数被调用一次,返回两次.子进程的返回值是0,父进程的返回值是子进程的进程id. 子进程和父进程继续执行fork调用之后的指令,子进程是父进程的副本,子进程获得父进程数据空间.堆和栈的副本.注意:这是子进程所拥有的副本,父子进程并不共享这些存储空间部分.父子进程共享正文段 #include <stdio.h> #include <sys/types.h> #include <unis

【APUE】wait与waitpid函数

当一个进程终止时,内核就向其父进程发送SIGCHLD信号.因为子进程终止是个异步事件,所以这种信号也是内核向父进程发的异步通知.父进程可以选择忽略该信号,或者提供一个该信号发生时即被调用执行的函数.对于这种信号的系统默认动作是忽略它. 调用wait或waitpid的进程发生的情况如下: 1.如果所有子进程都还在运行,则阻塞 2.如果一个子进程已终止,正等待父进程获取其终止状态,则取得该子进程的终止状态立即返回 3.如果它没有任何子进程,则立即出错返回 #include <sys/wait.h>

【APUE】进程间通信之管道

管道是UNIX系统IPC最古老形式,并且所有UNIX系统都提供此种通信机制.管道由下面两种局限性: 1)历史上,它们是半双工的(即数据只能在一个方向上流动) 2)它们只能在具有公共祖先的进程之间使用.通常,一个管道由一个进程创建,然后该进程调用fork,此后父.子进程之间就可应用该管道 管道由调用pipe函数创建: #include <unistd.h> int pipe(int filedes[2]);//若成功则返回0,出错返回-1 注意:filedes[0]为读而打开,filedes[1

【APUE】用户态与内核态的区别

当一个任务(进程)执行系统调用而陷入内核代码中执行时,我们就称进程处于内核运行态(或简称为内核态).此时处理器处于特权级最高的(0级)内核代码中 执行.当进程处于内核态时,执行的内核代码会使用当前进程的内核栈.每个进程都有自己的内核栈.当进程在执行用户自己的代码时,则称其处于用户运行态(用 户态).即此时处理器在特权级最低的(3级)用户代码中运行. 内核态与用户态是操作系统的两种运行级别,跟intel cpu没有必然的联系, intel cpu提供Ring0-Ring3三种级别的运行模式,Rin

【APUE】vim常用命令

转自:http://coolshell.cn/articles/5426.html 基本命令: i → Insert 模式,按 ESC 回到 Normal 模式. x → 删当前光标所在的一个字符. :wq → 存盘 + 退出 (:w 存盘, :q 退出)   (陈皓注::w 后可以跟文件名) dd → 删除当前行,并把删除的行存到剪贴板里 p → 粘贴剪贴板 简单的移动光标 0 → 数字零,到行头 ^ → 到本行第一个不是blank字符的位置(所谓blank字符就是空格,tab,换行,回车等)