进程.docx

上传人:b****7 文档编号:10571223 上传时间:2023-02-21 格式:DOCX 页数:10 大小:23.34KB
下载 相关 举报
进程.docx_第1页
第1页 / 共10页
进程.docx_第2页
第2页 / 共10页
进程.docx_第3页
第3页 / 共10页
进程.docx_第4页
第4页 / 共10页
进程.docx_第5页
第5页 / 共10页
点击查看更多>>
下载资源
资源描述

进程.docx

《进程.docx》由会员分享,可在线阅读,更多相关《进程.docx(10页珍藏版)》请在冰豆网上搜索。

进程.docx

进程

进程

1.父子进程如何通信?

一对父子进程可以通过正常的进程间通信的办法(管道,套接字,消息队列,共享内存)进行通信,但也可以通过利用它们作为父子进程的相互关系而具有的一

些特殊方法。

一个最显然的方法是父进程可以得到子进程的退出状态。

因为子进程从它的父进程继承文件描述符,所以父进程可以打开一个管道的两端,然后fork,然后父进程关闭管道这一端,子进程关闭管道另一端。

这正是你从你的进程调用‘popen()’函数运行另一个程序所发生的情况,也就是说你可以向‘popen()’返回的文件描述符进行写操作而子进程将其当作自己的标准输入,或者你可以读取这个文件描述符来看子进程向标准输出写了什么。

(‘popen()’函数的mode参数定义你的意图(译者注:

mode=“r”为读,mode=“w”为写);如果你想读写都做,那么你可以并不困难地用管道自己做到。

而且,子进程继承由父进程用mmap函数映射的匿名共享内存段(或者通过映射特殊文件‘/dev/zero’);这些共享内存段不能从无关的进程访问。

2.我怎样去除僵死进程?

2.1何为僵死进程?

当一个程序创建的子进程比父进程提前结束,内核仍然保存一些它的信息以便父进程会需要它-比如,父进程可能需要检查子进程的退出状态。

为了得到这些信息,父进程调用‘wait()’;当这个调用发生,内核可以丢弃这些信息。

在子进程终止后到父进程调用‘wait()’前的时间里,子进程被称为‘僵死进程’(‘zombie’)。

(如果你用‘ps’,这个子进程会有一个‘Z’出现在它的状态区里指出这点。

)即使它没有在执行,它仍然占据进程表里一个位置。

(它不消耗其它资源,但是有些工具程序会显示错误的数字,比如中央处理器的使用;这是因为为节约空间进程表的某些部份与会计数据(accountinginfo)是共用(overlaid)的。

这并不好,因为进程表对于进程数有固定的上限,系统会用光它们。

即使系统没有用光,每一个用户可以同时执行的进程数有限制,它总是小于系统的限制。

顺便说一下,这也正是你需要总是检查‘fork()’是否失败的一个原因。

如果父进程未调用wait函数而终止,子进程将被‘init’进程收管,它将控制子进程退出后必须的清除工作。

(‘init’是一个特殊的系统程序,进程号为1-它实际上是系统启动后运行的第一个程序),

2.2我怎样避免它们的出现?

你需要却认父进程为每个子进程的终止调用‘wait()’(或者‘waitpid()’,wait3()’,等等);或者,在某些系统上,你可以指令系统你对子进程的退出状

态没有兴趣。

(译者注:

在SysV系统上,可以调用signal函数,设置SIGCLD信号为SIG_IGN,系统将不产生僵死进程,详细说明参见<<高级编程>>10.7节)

另一种方法是*两次*‘fork()’,而且使紧跟的子进程直接退出,这样造成孙子进程变成孤儿进程(orphaned),从而init进程将负责清除它。

欲获得做这个的程序,参看范例章节的函数‘fork2()’。

为了忽略子进程状态,你需要做下面的步骤(查询你的系统手册页以知道这是否正常工作):

structsigactionsa;

sa.sa_handler=SIG_IGN;

#ifdefSA_NOCLDWAIT

sa.sa_flags=SA_NOCLDWAIT;

#else

sa.sa_flags=0;

#endif

sigemptyset(&sa.sa_mask);

sigaction(SIGCHLD,&sa,NULL);

如果这是成功的,那么‘wait()’函数集将不再正常工作;如果它们中任何一个被调用,它们将等待直到*所有*子进程已经退出,然后返回失败,并且

‘errno==ECHILD’。

另一个技巧是捕获SIGCHLD信号,然后使信号处理程序调用‘waitpid()’或‘wait3()’。

参见范例章节的完整程序。

3我怎样使我的程序作为守护程序运行?

一个“守护程序”进程通常被定义为一个后台进程,而且它不属于任何一个终端会话,(terminalsession)。

许多系统服务由守护程序实施;如网络服务,打印等。

简单地在后台启动一个程序并非足够是这些长时间运行的程序;那种方法没有正确地将进程从启动它的终端脱离(detach)。

而且,启动守护程序的普遍接受的的方法是简单地手工执行或从rc脚本程序执行(译者注:

rc:

runcom);并希望这个守护程序将其*自身*安置到后台。

这里是成为守护程序的步骤:

1.调用‘fork()’以便父进程可以退出,这样就将控制权归还给运行你程序的命令行或shell程序。

需要这一步以便保证新进程不是一个进程组头领进程(processgroupleader)。

下一步,‘setsid()’,会因为你是进程组头领进程而失败。

2.调用‘setsid()’以便成为一个进程组和会话组的头领进程。

由于一个控制终端与一个会话相关联,而且这个新会话还没有获得一个控制终端,我们的进程没有控制终端,这对于守护程序来说是一件好事。

3.再次调用‘fork()’所以父进程(会话组头领进程)可以退出。

这意味着我们,一个非会话组头领进程永远不能重新获得控制终端。

4.调用‘chdir("/")’确认我们的进程不保持任何目录于使用状态。

不做这个会导致系统管理员不能卸装(umount)一个文件系统,因为它是我们的当前工作目录。

[类似的,我们可以改变当前目录至对于守护程序运行重要的文件所在目录]

5.调用‘umask(0)’以便我们拥有对于我们写的任何东西的完全控制。

我们不知道我们继承了什么样的umask。

[这一步是可选的](译者注:

这里指步骤5,因为守护程序不一定需要写文件)

6.调用‘close()’关闭文件描述符0,1和2。

这样我们释放了从父进程继承的标准输入,标准输出,和标准错误输出。

我们没办法知道这些文描述符符可能

已经被重定向去哪里。

注意到许多守护程序使用‘sysconf()’来确认‘_SC_OPEN_MAX’的限制。

‘_SC_OPEN_MAX’告诉你每个进程能够打开的最多文件数。

然后使用一个循环,守护程序可以关闭所有可能的文件描述符。

你必须决定你需要做这个或不做。

如果你认为有可能有打开的文件描述符,你需要关闭它们,因为系统有一个同时打开文件数的限制。

7.为标准输入,标准输出和标准错误输出建立新的文件描述符。

即使你不打算使用它们,打开着它们不失为一个好主意。

准确操作这些描述符是基于各自爱好;比如说,如果你有一个日志文件,你可能希望把它作为标准输出和标准错误输出打开,而把‘/dev/null’作为标准输入打开;作为替代方法,你可以将‘/dev/console’作为标准错误输出和/或标准输出打开,而‘/dev/null’作为标准输入,或者任何其它对你的守护程序有意义的结合方法。

(译者注:

一般使用dup2函数原子化关闭和复制文件描述符,参见<<高级编程>>3.12节)

如果你的守护程序是被‘inetd’启动的,几乎所有这些步骤都不需要(或不建议采用)。

在那种情况下,标准输入,标准输出和标准错误输出都为你指定为网络连接,而且‘fork()’的调用和会话的操纵不应做(以免使‘inetd’造成混乱)。

只有‘chdir()’和‘umask()’这两步保持有用。

4我怎样象ps程序一样审视系统的进程?

你真的不该想做这个。

到目前为止,移植性最好的是调用‘popen(pscmd,"r")’并处理它的输出。

(pscmd应当是类似SysV系统上的‘“ps-ef”’,BSD系统有很多可能的显示选项:

选择一个。

在范例章节有这个问题的两个完整解决方法;一个适用于SunOS4,它需要root权限执行并使用‘kvm_*’例程从内核数据结果读取信息;另一种适用于SVR4系统(包括SunOS5),它使用‘/proc’文件系统。

在具有SVR4.2风格‘/proc’的系统上更简单;只要对于每一个感兴趣的进程号从文件‘/proc/进程号/psinfo’读取一个psinfo_t结构。

但是,这种可能是最清晰的方法也许又是最不得到很好支持的方法。

(在FreeBSD的‘/proc’上,你从‘/proc/进程号/status’读取一个半未提供文档说明(semi-undocumented)的可打印字符串;Linux有一些与其类似的东西)

5给定一个进程号,我怎样知道它是个正在运行的程序?

使用‘kill()’函数,而已0作为信号代码(signalnumber)。

从这个函数返回有四种可能的结果:

*‘kill()’返回0

-这意味着一个给定此进程号的进程退出,系统允许你向它发送信号。

该进

程是否可以是僵死进程与不同系统有关。

*‘kill()’返回-1,‘errno==ESRCH’

-要么不存在给定进程号的进程,要么增强的安全机制导致系统否认它的存

在。

(在一些系统上,这个进程有可能是僵死进程。

*‘kill()’返回-1,‘errno==EPERM’

-系统不允许你杀死(kill)这个特定进程。

这意味着要么进程存在(它又可能是

僵死进程),要么严格的增强安全机制起作用(比如你的进程不允许发送信号

给*任何人*)。

*‘kill()’返回-1,伴以其它‘errno’值

-你有麻烦了!

用的最多的技巧是认为调用“成功”或伴以‘EPERM’的“失败”意味着进程存在,而其它错误意味着它不存在。

如果你特别为提供‘/proc’文件系统的系统(或所有类似系统)写程序,一个替换方法存在:

检查‘proc/进程号’是否存在是可行的。

6system函数,pclose函数,waitpid函数的返回值是什么?

‘system()’,‘pclose()’或者‘waitpid()’的返回值不象是我进程的退出值(exitvalue)(译者注:

退出值指调用exit()或_exit()时给的参数)...或者退出值左移了8位...这是怎么搞的?

手册页是对的,你也是对的!

如果查阅手册页的‘waitpid()’你会发现进程的返回值被编码了。

正常情况下,进程的返回值在高16位,而余下的位用来作其它事。

如果你希望可移植,你就不能凭借这个,而建议是你该使用提供的宏。

这些宏总是在‘wait()’或‘wstat’的文档中说明了。

为了不同目的定义的宏(在‘’)包括(stat是‘waitpid()’返回的值):

`WIFEXITED(stat)'

如果子进程正常退出则返回非0

`WEXITSTATUS(stat)'

子进程返回的退出码

`WIFSIGNALED(stat)'

如果子进程由与信号而终止则返回非0

`WTERMSIG(stat)'

终止子进程的信号代码

`WIFSTOPPED(stat)'

如果子进程暂停(stopped)则返回非0

`WSTOPSIG(stat)'

使子进程暂停的信号代码

`WIFCONTINUED(stat)'

如果状态是表示子进程继续执行则返回非0

`WCOREDUMP(stat)'

如果‘WIFSIGNALED(stat)’为非0,而如果这个进程产生一个内存映射文件(coredump)则返回非0

7我怎样找出一个进程的存储器使用情况?

如果提供的话,参看‘getrusage()’手册页

7.1为什么进程的大小不缩减?

当你使用‘free()’函数释放内存给堆时,几乎所有的系统都*不*减少你程序的对内存的使用。

被‘free()’释放的内存仍然属于进程地址空间的一部份,并将

被将来的‘malloc()’请求所重复使用。

如果你真的需要释放内存给系统,参看使用‘mmap()’分配私有匿名内存映射(privateanonymousmappings)。

当这些内存映射被取消映射时,内存真的将其释放给系统。

某些‘malloc()’的实现方法(比如在GNUC库中)在允许时自动使用‘mmap()’实施大容量分配;这些内存块(blocks)随着‘free()’被释放回系统。

当然,如果你的程序的大小增加而你认为它不应该这样,你可能有一个‘内存泄露’(‘memoryleak’)-即在你的的程序中有缺陷(bug)导致未用的内存没释放。

7.2我怎样改变我程序的名字(即“ps”看到的名字)?

在BSD风格的系统中,‘ps’程序实际上审视运行进程的地址空间从而找到当前的‘argv[]’,并显示它。

这使得程序可以通过简单的修改‘argv[]’以改变它的名字。

在SysV风格的系统中,命令的名字和参数的一般头80字节是存放在进程的u-区(u-area),所以不能被直接修改。

可能有一个系统调用用来修改它(不象是这样),但是其它的话,只有一个方法就是实施一个‘exec()’,或者些内核内存(危险,而且只有root才有可能)。

一些系统(值得注意的是Solaris)可以有‘ps’的两种不同版本,一种是在

‘/usr/bin/ps’拥有SysV的行为,而另一种在‘/usr/ucb/ps’拥有BSD的行为。

在这些系统中,如果你改变‘argv[]’,那么BSD版的‘ps’将反映这个变化,而SysV版将不会。

检查你的系统是否有一个函数‘setproctitle()’。

7.3我怎样找到进程的相应可执行文件?

这个问题可以作为‘常见未回答问题’(‘FrequentlyUnansweredQuestions’)的一个好候选,因为事实上提出这个问题经常意味着程序的设计有缺陷。

:

你能作的‘最佳猜测’(‘bestguess’)是通过审视‘argv[0]’的值而获得。

如果它包括一个‘/’,那么它可能是可执行程序的绝对或相对(对于在程序开始时的当前目录而言)路径。

如果不包括,那么你可以仿效shell对于‘PATH’变量的查询来查找这个程序。

但是,不能保证成功,因为有可能执行程序时‘argv[0]’是一些任意值,也不排除这个可执行文件在执行后可能已经被更名或删除的情况。

如果所有你想做的只是能打印一个和错误消息一起出现的合适的名字,那么最好的方法在‘main()’函数中将‘argv[0]’的值保存在全局变量中以供整个程序使用。

虽然没有保证说‘argv[0]’的值总是有意义,但在大多数情况下它是最好的选择。

人们询问这个问题的最普通原因是意图定位他们程序的配置文件。

这被认为是不好的形式;包含可执行文件的目录应当*只*包含可执行文件,而且基于管理的要求经常试图将配置文件放置在和可执行文件不同的文件系统。

试图做这个的一个比较不普通但更正规的理由是允许程序调用‘exec()’执行它自己;这是一种用来完全重新初始化进程(比如被用于一些‘sendmail’的版本)的办法(比如当一个守护程序捕获一个‘SIGHUP’信号)。

7.4那么,我把配置文件放在哪里里呢?

为配置文件安排正确的目录总是取决于你使用的Unix系统的特点;

‘/var/opt/PACKAGE’,‘/usr/local/lib’,‘/usr/local/etc’,或者任何其它一

些可能的地方。

用户自定义的配置文件通常是在‘$HOME’下的以“.”开始的隐藏文件(比如‘$HOME/.exrc’)。

从一个在不同系统上都能使用的软件包(package)的角度看,它通常意味着任何站点范围(sitewide)的配置文件的位置有个已设定的缺省值,可能情况是使用一个在配置脚本程序里的‘--prefix’选项(Autoconf脚本程序集做这个工作)。

你会希望允许这个缺省值在程序执行时被一个环境变量重载。

(如果你没使用配置脚本程序,那么在编译时,将这个位置缺省值作为‘-D’选项放入项目文件(Makefile),或者将其放入一个‘config.h’头文件,或做其它类似的工作)

用户自定义配置需要放置于一个在‘$HOME’下的文件名“.”打头的文件,或者在需要多个配置文件时,建立文件名“.”打头的子目录。

(在列目录时,文件名以“.”打头的文件或目录缺省情况下被忽略。

)避免在‘$HOME’建立多个文件,因为这会造成非常杂乱的情况。

当然,你也应该允许用户通过一个环境变量重载这个位置。

即使不能找到某个用户的配置文件,程序仍应当以适宜的方式执行。

7.5为何父进程死时,我的进程未得到SIGHUP信号?

因为本来就没有设想是这样做的。

‘SIGHUP’是一个信号,它按照惯例意味着“终端线路被挂断”。

它与父进程无关,而且通常由tty驱动程序产生(并传递给前台的进程组)。

但是,作为会话管理系统(sessionmanagementsystem)的一部份,确切说有两种情况下‘SIGHUP’会在一个进程死时发送出:

*当一个终端设备与一个会话相关联,而这个会话的会话首领进程死时,‘SIGHUP’被发送至这个终端设备的所有前台进程组。

*当一个进程死去导致一个进程组变成孤儿,而且该进程组里一个或多个进程处于*暂停*状态时,那么‘SIGHUP’和‘SIGCONT’被发送至这个孤儿进程

组的所有成员进程。

(一个孤儿进程组是指在该进程组中没有一个成员进程的父进程属于和该进程组相同的会话的其它进程组。

7.6我怎样杀死一个进程的所有派生进程?

没有一个完全普遍的方法来做这个。

虽然你可以通过处理‘ps’的输出确定进程间的相互关系,但因为它只表示系统的一瞬间的状态(snapshot)所以并不可靠。

但是,如果你启动一个子进程,而它可能生成它自己的子进程,而你意图一次杀死整个生成的事务(job),解决方法是将最先启动的子进程置于一个新的进程组,当你需要时杀死整个进程组。

建议为创建进程组而使用的函数是‘setpgid()’。

在可能情况下,使用这个函数而不使用‘setpgrp()’,因为后一个在不同系统中有所不同(在一些系统上‘setgrp();’等同于‘setpgid(0,0);’,在其它系统上,‘setpgrp()’和‘setpgid()’相同)。

参见范例章节的事务-控制范例程序。

放置一个子进程于其自身的进程组有一些影响。

特别的,除非你显式地将该进程组放置于前台,它将被认为是一个后台事务并具有以下结果:

*试图从终端读取的进程将被‘SIGTTIN’信号暂停。

*如果设置终端模式‘tostop’,那么试图向终端写的进程将被‘SIGTTOU’信号暂停。

(试图改变终端模式也导致这个结果,且不管当前‘tostop’是否设置)

*子进程将不会收到从终端发出的键盘信号(比如‘SIGINT’或‘SIGQUIT’)在很多应用程序中输入和输出总会被重定向,所以最显著的影响将是丧失键盘信号。

父进程需要安排程序起码捕获‘SIGINIT’和‘SIGQUIT’(可能情况下,还有‘SIGTERM’),并在需要情况下清除后台事务。

展开阅读全文
相关资源
猜你喜欢
相关搜索

当前位置:首页 > 高等教育 > 哲学

copyright@ 2008-2022 冰豆网网站版权所有

经营许可证编号:鄂ICP备2022015515号-1