Chinaunix首页 | 论坛 | 博客
  • 博客访问: 61969
  • 博文数量: 21
  • 博客积分: 1410
  • 博客等级: 上尉
  • 技术积分: 210
  • 用 户 组: 普通用户
  • 注册时间: 2008-04-08 10:48
个人简介

完美之道,不在无可增加,而在无可删减。

文章分类
文章存档

2014年(2)

2008年(19)

我的朋友

分类: LINUX

2008-04-21 18:35:37

fork vfork exec wait使用

vfork用于创建一个新进程,而该新进程的目的是exec一个新进程,vforkfork一样都创建一个子进程,但是它并不将父进程的地址空间完全复制到子进程中,不会复制页表。因为子进程会立即调用exec,于是也就不会存放该地址空间。不过在子进程中调用execexit之前,他在父进程的空间中运行。

为什么会有vfork,因为以前的fork当它创建一个子进程时,将会创建一个新的地址空间,并且拷贝父进程的资源,而往往在子进程中会执行exec调用,这样,前面的拷贝工作就是白费力气了,这种情况下,聪明的人就想出了vfork,它产生的子进程刚开始暂时与父进程共享地址空间(其实就是线程的概念了),因为这时候子进程在父进程的地址空间中运行,所以子进程不能进行写操作,并且在儿子霸占着老子的房子时候,要委屈老子一下了,让他在外面歇着(阻塞),一旦儿子执行了exec或者exit后,相当于儿子买了自己的房子了,这时候就相当于分家了。

vforkfork之间的另一个区别是: vfork保证子进程先运行,在她调用execexit之后父进程才可能被调度运行。如果在调用这两个函数之前子进程依赖于父进程的进一步动作,则会导致死锁。

由此可见,这个系统调用是用来启动一个新的应用程序。其次,子进程在vfork()返回后直接运行在父进程的栈空间,并使用父进程的内存和数据。这意味着子进程可能破坏父进程的数据结构或栈,造成失败。

为了避免这些问题,需要确保一旦调用vfork(),子进程就不从当前的栈框架中返回,并且如果子进程改变了父进程的数据结构就不能调用exit函数。子进程还必须避免改变全局数据结构或全局变量中的任何信息,因为这些改变都有可能使父进程不能继续。

通常,如果应用程序不是在fork()之后立即调用exec(),就有必要在fork()被替换成vfork()之前做仔细的检查。

fork函数创建子进程后,子进程往往要调用一种exec函数以执行另一个程序,当进程调用一种exec函数时,该进程完全由新程序代换,而新程序则从其main函数开始执行,因为调用exec并不创建新进程,所以前后的进程id 并未改变,exec只是用另一个新程序替换了当前进程的正文,数据,堆和栈段。

以下程序演示的是使用vfork后,让父进程一直等待之进程执行完或是异常推出之后才执行的例子。可以成子谓杀不死的子进程。

/*

*main.c

*just do while(1);

*/

#include

int main()

{

    do

    {}

    while(1);

   

    return 0;

}

 

/*

*vfork.c

*let child do his job

*father will do his after child died

*/

#include

#include

#include

#include

 

int main()

{

int fd=0;

reforkfd=vfork();

if(fd<0)

perror(“vfork”);

if(fd==0)

{

Printf(“child!\n”);

execl(“./main”,”main”,NULL);

exit(0);

}

else

{

wait(0);

printf(“father!\n”);

goto refork

}

return 0;

}

阅读(1721) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~