在前面学习了进程的创建、等待、退出之后,我们知道了在有些时候子进程和父进程会共享地址空间,比如在Vfork()函数调用的时候,所以,今天我们来学习一下进程替换和一个简单的小程序。
1、进程程序替换
(1)替换原理
用fork创建子进程后执行的是和父进程相同的程序(但有可能执行不同的代码分支),子进程往往要调用一种exec函数来以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新程序替换,从新程序的启动例程开始执行。调用exec函数并不创建新程序,所以调用exec函数前后该进程的id并未改变。
(2)替换函数
替换函数其实有六种,都是以exec开头的函数,统称为exec函数
(3)函数解释
这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回;
如果调用出错则返回-1;
所有exec函数只有出错的返回值,没有成功的返回值;
(4)命名理解
- l(list):表示参数采用列表;
- v(vector):参数用数组;
- p(path):有P自动搜索环境变量PATH;
- e(env):表示自己维护环境变量;
2、简单实例之实现简单的shell
要想编写简单的shell,我们就要了解shell收取到一个命令后的运行过程,接下来就以一个简单的例子来说明一下;
首先我们先考虑一下下面这个shell实现命令的过程:
我们先来分析一下,shell从用户读入字符串ls,然后,shell建立一个新的进程,然后再那个进程中运行ls程序并等待那个进程结束,然后再读取下一个字符串.......
然后shell读取新的一行输入,建立一个新的进程,在这个进程中运行程序,并等待该进程退出。
所以,要想实现一个简单的shell,大致思想就是对一下步骤进行循环:
- 获取命令行;
- 解析命令行;
- 建立一个子进程(fork);
- 替换子进程(execvp);
- 父进程等待子进程退出;
接下来,我们就来用代码实现一个简单的shell。
#include <unistd.h>
#include<sys/types.h>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
char *argv[8];
int argc=0;
void do_parse(char *buf)
{
int i=0;
int status=0;
for (argc=i=0;buf[i];i++)
{
if(!isspace(buf[i])&&status==0)
{
argv[argc++]=buf+i;
status=1;
}
else if(isspace(buf[i]))
{
status=0;
buf[i]=0;
}
}
argv[argc]=NULL;
}
void do_execute()
{
pid_t pid=fork();
switch(pid)
{
case -1:perror("fork");
exit(EXIT_FAILURE);
break;
case 0:execvp(argv[0],argv);
perror("execvp");
exit(EXIT_FAILURE);
break;
default:{
int st;
while(wait(&st)!=pid)
;
}
}
}
int main()
{
char buf[1024]={};
while(1)
{
printf("[myshell]:> ");
scanf("%[^\n]%*c",buf);
do_parse(buf);
do_execute();
}
}
简单的shell就实现完成,接下来,我们就来使用一下我们的myshell:
好了,对于进程今天我们就先学到这。多有不足,望大佬们多指点。