Version | S | Description | Date | By |
---|---|---|---|---|
V1 | C | First Version | 2020-12-10 | AYZP |
C―― Create,
A—— Add,
M—— Modify,
D—— Delete。
前言
在操作系统的学习中,有一个fork函数,简单记录下它的点点,慢慢补充。
一 我咧fork
1.1 功能
fork:从父进程创建子进程。
fork成功的话,父进程就会得到子进程的pid。
1.2 过程
fork,先从父进程拷贝出来一个父进程的副本,然后通过调用exec文件载入其他程序,子进程和父进程此时就可以运行不同的程序。在子进程状态还未完成时,父进程调用wait进行阻塞一直等待子进程的状态改变。
#include <stdio.h>
void main(int argc, char *argv[])
{
int pid;
pid = fork(); // fork another process
if (fork < 0) //error occurred
{
fprintf(srderr, "Fork Failed");
exit (-1);
}
else if (fork == 0) // child process, CPU在子进程中
{
execlp("/bin/ls", "ls", NULL);
}
else // parent process,CPU在父进程中
{
wait(NULL);
printf("Child Complete");
exit(0);
}
}
// fork成功后,系统中就会有两个进程,一个父进程,一个父进程创建的子进程
二 参考回答:
Fork:创建一个和当前进程映像一样的进程可以通过fork( )系统调用:
#include <sys/types.h>
#include <unistd.h>
pid_t fork(void);
成功调用fork( )会创建一个新的进程,它几乎与调用fork( )的进程一模一样,这两个进程都会继续运行。在子进程中,成功的fork( )调用会返回0。在父进程中fork( )返回子进程的pid。如果出现错误,fork( )返回一个负值。
最常见的fork( )用法是创建一个新的进程,然后使用exec( )载入二进制映像,替换当前进程的映像。这种情况下,派生(fork)了新的进程,而这个子进程会执行一个新的二进制可执行文件的映像。这种“派生加执行”的方式是很常见的。
在早期的Unix系统中,创建进程比较原始。当调用fork时,内核会把所有的内部数据结构复制一份,复制进程的页表项,然后把父进程的地址空间中的内容逐页的复制到子进程的地址空间中。但从内核角度来说,逐页的复制方式是十分耗时的。现代的Unix系统采取了更多的优化,例如Linux,采用了写时复制的方法,而不是对父进程空间进程整体复制。
三 其他问题
3.1 请你说说fork,wait,exec函数
父进程产生子进程使用fork拷贝出来一个父进程的副本,此时只拷贝了父进程的页表,两个进程都读同一块内存,当有进程写的时候使用写实拷贝机制分配内存,exec函数可以加载一个elf文件去替换父进程,从此父进程和子进程就可以运行不同的程序了。fork从父进程返回子进程的pid,从子进程返回0.调用了wait的父进程将会发生阻塞,直到有子进程状态改变,执行成功返回0,错误返回-1。exec执行成功则子进程从新的程序开始运行,无返回值,执行失败返回-1
参考文献
[1] 田丽华. 操作系统原理. 中国大学MOOC. 2020.
https://www.icourse163.org/course/XJTU-1003409001