fork和exec有什么区别?
当前回答
Fork()将当前进程拆分为两个进程。换句话说,你的线性程序突然变成了两个单独的程序,运行着一段代码:
int pid = fork();
if (pid == 0)
{
printf("I'm the child");
}
else
{
printf("I'm the parent, my child is %i", pid);
// here we can kill the child, but that's not very parently of us
}
这可能会让你大吃一惊。现在,两个进程执行了一段状态几乎相同的代码。子进程继承刚刚创建它的进程的所有代码和内存,包括从fork()调用刚刚停止的地方开始。唯一的区别是fork()返回的代码告诉你你是父类还是子类。如果你是父进程,返回值是子进程的id。
Exec更容易掌握一些,您只需告诉Exec使用目标可执行文件执行一个进程,并且不需要两个进程运行相同的代码或继承相同的状态。就像@Steve Hawkins说的,exec可以在你fork后在当前进程中执行目标可执行文件。
其他回答
它们一起使用来创建一个新的子进程。首先,调用fork创建当前进程(子进程)的副本。然后,在子进程内部调用exec,用新进程“替换”父进程的副本。
这个过程是这样的:
child = fork(); //Fork returns a PID for the parent process, or 0 for the child, or -1 for Fail
if (child < 0) {
std::cout << "Failed to fork GUI process...Exiting" << std::endl;
exit (-1);
} else if (child == 0) { // This is the Child Process
// Call one of the "exec" functions to create the child process
execvp (argv[0], const_cast<char**>(argv));
} else { // This is the Parent Process
//Continue executing parent process
}
Fork创建调用进程的副本。 一般遵循结构
int cpid = fork( );
if (cpid = = 0)
{
//child code
exit(0);
}
//parent code
wait(cpid);
// end
(对于子进程文本(代码),数据,堆栈与调用进程相同) 子进程执行if块中的代码。
EXEC用新进程的代码、数据和堆栈替换当前进程。 一般遵循结构
int cpid = fork( );
if (cpid = = 0)
{
//child code
exec(foo);
exit(0);
}
//parent code
wait(cpid);
// end
(exec调用后,Unix内核清除子进程的文本,数据,堆栈,并填充与foo进程相关的文本/数据) 因此子进程使用不同的代码(foo的代码{与父进程不相同})
理解fork()和exec()概念的主要例子是shell,用户通常在登录到系统后执行命令解释器程序。shell将命令行的第一个单词解释为命令名
对于许多命令,shell fork和子进程执行与名称相关的命令,将命令行上的剩余单词作为命令的参数。
shell允许三种类型的命令。首先,命令可以是 包含由源代码编译产生的目标代码的可执行文件(例如C程序)。其次,命令可以是一个可执行文件 包含shell命令行序列。最后,命令可以是一个内部shell命令。(而不是一个可执行文件ex->cd,ls等)
fork ():
它创建正在运行的进程的副本。正在运行的进程称为父进程,新创建的进程称为子进程。区分两者的方法是通过查看返回值:
Fork()返回父进程中子进程的标识符(pid) Fork()在子对象中返回0。
exec ():
它在一个流程中启动一个新流程。它将一个新程序加载到当前进程中,替换现有的程序。
Fork () + exec():
当启动一个新程序时,首先fork(),创建一个新进程,然后exec()(即加载到内存并执行)它应该运行的程序二进制。
int main( void )
{
int pid = fork();
if ( pid == 0 )
{
execvp( "find", argv );
}
//Put the parent to sleep for 2 sec,let the child finished executing
wait( 2 );
return 0;
}
Fork()将当前进程拆分为两个进程。换句话说,你的线性程序突然变成了两个单独的程序,运行着一段代码:
int pid = fork();
if (pid == 0)
{
printf("I'm the child");
}
else
{
printf("I'm the parent, my child is %i", pid);
// here we can kill the child, but that's not very parently of us
}
这可能会让你大吃一惊。现在,两个进程执行了一段状态几乎相同的代码。子进程继承刚刚创建它的进程的所有代码和内存,包括从fork()调用刚刚停止的地方开始。唯一的区别是fork()返回的代码告诉你你是父类还是子类。如果你是父进程,返回值是子进程的id。
Exec更容易掌握一些,您只需告诉Exec使用目标可执行文件执行一个进程,并且不需要两个进程运行相同的代码或继承相同的状态。就像@Steve Hawkins说的,exec可以在你fork后在当前进程中执行目标可执行文件。