- 学习进程创建, 等待, 终止. 使用代码实现.
- 编写自主shell.
- 封装fork/wait等操作, 编写函数 process_create(pid_t* pid, void* func, void* arg), func回调函数就是子进程执行的入口函数, arg是传递给func回调函数的参数.
- 调研popen/system, 理解这两个函数和fork的区别.
写一篇博客, 总结上述内容. 作业以链接形式提交(代码也是在博客中体现).
进程创建
fork函数:
在linux中fork函数从已存在进程中创建一个新进程。新进程为子进程,而原进程为父进程。
#include <unistd.h>
pid_t fork(void);
//f返回值:子进程中返回0,父进程中返回子进程id,出错返回-1
进程调用fork,当控制转移到内核中的fork代码后,内核做:
- 分配新的内存块和内核数据结构给子进程
- 将父进程部分数据结构内容拷贝至子进程
- 添加子进程到系统进程列表当中
- fork返回,开始调度器调度
当一个进程调用fork之后,就有两个二进制代码相同的进程。而且他们都运行到相同的地方。但每一个进程都可以可是他们自己的旅程,看下面程序
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <string.h>
#include <errno.h>
int main()
{
pid_t pid;
printf("before fork pid = %d\n", getpid());
pid = fork();
if(pid == -1)
{
perror("errorno");
exit(EXIT_FAILURE);
}else if(pid == 0)
{
int i = 0;
while(i<20)
{
printf("Child:\npid = %d ppid = %d\n\n",getpid(),getppid());
sleep(1);
i++;
}
}else if(pid > 0 )
{
int i = 0;
while(i<20)
{
printf("Parent:\n pid = %d,cpid = %d\n",getpid(),pid);
sleep(1);
i++;
}
}
return 0;
}
运行结果
方法二:
pid_vfork() //创建一个子进程,并阻塞父进程
特点:
1.父进程阻塞,直到子进程运行完毕
2.就算写时,也不拷贝,也就是父子进程共享数据段、代码段所有内存空间
3.必须使用exit(0)或exec(0)返回
4.每个系统对vfork的实现或多或少都有问题,不要使用。
进程终止
进程退出场景:
- 代码运行完毕,结果正确
- 代码运行完毕,结果不正确
- 代码异常终止
销毁进程的过程:
1.释放资源。内存,文件等等
2.记账信息
3.将进程设置成僵尸状态
4.转存储调度。将CPU让给需要使用的进程。
进程退出的方法:
正常退出:
1.main函数的退出
2.exit
异常退出:
1. Ctrl + C
2. abort
3. kill (给进程发送一个信号,三种不正常退出实质上都是给进程发送信号)
exit函数
#include <unistd.h>
void exit(int status);
exit 最后也会调用_exit,但在调用exit之前,还做了其他工作:
1. 执行用户通过atexit或on_exit定义的清理函数。
2. 关闭所有打开流,所有的缓存数据均被写入
3. 调用_exit