Linux进程控制(三)

时间:2020-12-10 19:51:44

1. 进程间打开文件的继承

1.1. 用fork继承打开的文件

fork以后的子进程自动继承了父进程的打开的文件,继承以后,父进程关闭打开的文件不会对子进程造成影响。

示例:

#include <stdio.h>

#include <fcntl.h>

#include <unistd.h>

#include <sys/types.h>

#include <sys/stat.h>

int main()

{

char szBuf[32] = {'\0'};

int iFile = open("./a.txt", O_RDONLY);

if(fork() > 0){//parent process

close(iFile);

return 0;

}

//child process

sleep(3); //wait for parent process closing fd

if(read(iFile, szBuf, sizeof(szBuf)-1) < 1){

perror("read fail");

}else{

printf("string:%s\n",szBuf);

}

close(iFile);

return 0;

}

1.2. 守护进程

Daemon运行在后台也称作“后台服务进程”。 它是没有控制终端与之相连的进程。它独立与控制终端、通常周期的执行某种任务。那么为什么守护进程要脱离终端后台运行呢?守护进程脱离终端是为了避免进程在执行过程中的信息在任何终端上显示并且进程也不会被任何终端所产生的任何终端信息所打断。那么为什么要引入守护进程呢?由于在linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依赖这个终端,这个终端就称为这些进程的控制终端。当控制终端被关闭时,相应的进程都会自动关闭。但是守护进程却能突破这种限制,它被执行开始运转,直到整个系统关闭时才退出。几乎所有的服务器程序如Apache和wu-FTP,都用daemon进程的形式实现。很多Linux下常见的命令如inetd和ftpd,末尾的字母d通常就是指daemon。

守护进程的特性:

1> 守护进程最重要的特性是后台运行。

2> 其次,守护进程必须与其运行前的环境隔离开来。这些环境包括未关闭的文件描述符、控制终端、会话和进程组、工作目录已经文件创建掩码等。这些环境通常是守护进程从父进程那里继承下来的。

3> 守护进程的启动方式

daemon进程的编程规则

l 创建子进程,父进程退出:

调用fork产生一个子进程,同时父进程退出。我们所有后续工作都在子进程中完成。这样做我们可以交出控制台的控制权,并为子进程作为进程组长作准备;由于父进程已经先于子进程退出,会造成子进程没有父进程,变成一个孤儿进程(orphan)。每当系统发现一个孤儿进程,就会自动由1号进程收养它,这样,原先的子进程就会变成1号进程的子进程。代码如下:

pid = fork();

if(pid>0)

exit(0);

l 在子进程中创建新会话:

使用系统函数setsid()。由于创建守护进程的第一步调用了fork函数来创建子进程,再将父进程退出。由于在调用fork函数的时候,子进程全盘拷贝了父进程的会话期、进程组、控制终端等,虽然父进程退出了,但会话期、进程组、控制终端并没有改变,因此,还不是真正意义上的独立开来。而调用setsid函数会创建一个新的会话并自任该会话的组长,调用setsid函数有下面3个作用:让进程摆脱原会话的控制,让进程摆脱原进程组的控制,让进程摆脱原控制终端的控制;

进程组:是一个或多个进程的集合。进程组有进程组ID来唯一标识。除了进程号(PID)之外,进程组ID(GID)也是一个进程的必备属性。每个进程都有一个组长进程,其组长进程的进程号等于进程组ID。且该进程组ID不会因为组长进程的退出而受影响。

会话周期:会话期是一个或多个进程组的集合。通常,一个会话开始于用户登录,终止于用户退出,在此期间该用户运行的所有进程都属于这个会话期。

控制终端:由于在linux中,每一个系统与用户进行交流的界面称为终端,每一个从此终端开始运行的进程都会依赖这个控制终端。

l 改变当前目录为根目录:

使用fork函数创建的子进程继承了父进程的当前工作目录。由于在进程运行中,当前目录所在的文件是不能卸载的,这对以后的使用会造成很多的不便。利用chdir("/");把当前工作目录切换到根目录。

l 重设文件权限掩码:

umask(0);将文件权限掩码设为0,Deamon创建文件不会有太大麻烦;

l 关闭所有不需要的文件描述符:

新进程会从父进程那里继承一些已经打开了的文件。这些被打开的文件可能永远不会被守护进程读写,而它们一直消耗系统资源。另外守护进程已经与所属的终端失去联系,那么从终端输入的字符不可能到达守护进程,守护进程中常规方法(如printf)输出的字符也不可能在终端上显示。所以通常关闭从0到MAXFILE的所有文件描述符。

for(i=0;i<MAXFILE;i++)

close(i);

(注:有时还要处理SIGCHLD信号signal(SIGCHLD, SIG_IGN);防止僵尸进程(zombie))

下面就可以添加任何你要daemon做的事情

示例:

#include <stdio.h>

#include <stdlib.h>

#include <unistd.h>

#include <sys/stat.h>

void Daemon()

{

const int MAXFD=64;

int i=0;

if(fork()!=0) //父进程退出

exit(0);

setsid(); //成为新进程组组长和新会话领导,脱离控制终端

chdir("/"); //设置工作目录为根目录

umask(0); //重设文件访问权限掩码

for(;i<MAXFD;i++) //尽可能关闭所有从父进程继承来的文件

close(i);

}

int main()

{

Daemon(); //成为守护进程

while(1){

sleep(1);

}

return 0;

}

Example:

#include <stdio.h>

#include <stdlib.h>

#include <unistd.h>

#include <syslog.h>

#include <sys/stat.h>

#include <time.h>

main()

{

int i = 0;

if(fork() > 0)

exit(0);

setsid();

chdir("/");

umask(0);

for(; i < 64; i++)

{

close(i);

}

i = 0;

while(i < 10)

{

printf("%d\n",i);

time_t ttime;

time(&ttime);

struct tm *pTm = gmtime(&ttime);

syslog(LOG_INFO,"%d %04d:%02d:%02d", i, (1900 + pTm->tm_year), (1 + pTm->tm_mon), (pTm->tm_mday));

i++;

sleep(2);

}

}

通过查看vi /var/log/messages