进程程序替换
为什么要有进程替换
我们之前学到过,fork创建子进程,目的是让子进程执行和父进程相同的程序,并且父进程只能获得子进程某些方面的特殊信息。而我们期望的是子进程和父进程执行不同的程序,做不同的事情,所以就有了进程替换 。
替换原理
fork创建子进程后,子进程要调用一种exec函数以执行另一个程序。当进程调用一种exec函数时,该进程的用户空间代码和数据完全被新进程替换,从新程序的启动例程开始执行。调用exec并不创建新进程,所以调用exec前后进程的id并未改变。如图:
替换函数
有六种以exec开头的函数,统称为exec函数:
#include<unistd.h>
int execl(const char *path,const char *arg, ...);
int execlp(const char *file,const char *arg, ...);
int execle(const char *path,const char *arg, ...,char *const envp[]);
int execv(const char *path,char *const argv[]);
int execvp(const char * file,char *const argv[]);
int execve(const char *path,char *const argv[],char *const envp[]);
前5个函数都是对execve函数的封装,系统只调用execve,参数中的“...”表示可变参数列表。
函数解释
- 这些函数如果调用成功则加载新的程序从启动代码开始执行,不再返回。
- 如果调用出错则返回-1
- 所以exec函数只有出错的返回值而没有成功的返回值。
命名的理解
这些函数原型看起来比较相似很容易记混,但只要掌握规律就很好记。
- l(list):表示参数采用列表。
- v(vector):参数用数组。
- p(path):有p自动搜索环境变量PATH。
- e(env):表示自己维护环境变量。
exec调用示例:
#inlcude<unistd.h>
int main()
{
char *const argv[]={"ps","-ef",NULL};
char *const envp[]={"PATH=/bin:usr/bin","TERM=console",NULL};
execl("/bin/ps","ps","-ef",NULL);
//带p的,可以使用环境变量PATH,无需写全路径
execlp("ps", "ps", "-ef", NULL);
// 带e的,需要自己组装环境变量
execle("ps", "ps", "-ef", NULL, envp);
execv("/bin/ps", argv);
// 带p的,可以使用环境变量PATH,无需写全路径
execvp("ps", argv);
// 带e的,需要自己组装环境变量
execve("/bin/ps", argv, envp);
exit(0);
}
只有execve是真正的系统调用,其他5个函数最终都调用execve。这些函数之间的关系如下图所示:
综合前面所学的知识我们可以简单地实现一个shell
当一条命令输入到命令行时,shell会做一系列事情,如下图,时间轴表示事件的发生次序:
shell读取新的一行输入,建立一个新的进程,在这个进程中运行程序并等待这个进程结束。所以,要写一个shell,需要循环以下过程:
- 获取命令行
- 解析命令行
- 建立一个子进程(fork)
- 替换子进程(execvp)
- 父进程等待子进程退出(wait)
实现代码:
#include<stdio.h>
#include<unistd.h>
#include<string.h>
#include<stdlib.h>
#include<sys/types.h>
#include<sys/wait.h>
#define MAX_COM 1024
#define NUM 32
int main()
{
char buff[MAX_COM];
static char *argv[NUM];
while(1)
{
printf("[ty@localhost minshell]$ ");
fflush(stdout);
fgets(buff,MAX_COM,stdin);
buff[strlen(buff)-1]= 0;
//解析字符串
int argc= 0;
argv[argc]=strtok(buff," ");//拆分字符串
argc++;
while(1)
{
char *ptr=strtok(NULL," ");
if(ptr == NULL)
{
argv[argc]=NULL;
break;
}
argv[argc]=ptr;
argc++;
}
//执行命令
pid_t pid=fork();//创建子进程进行程序替换
if(pid< 0)
{
;//fork创建子进程失敗
}
else if(pid == 0)//child
{
execvp(argv[0],argv);
}
else//parent
{
waitpid(pid,NULL,0);
}
}
return 0;
}
程序运行起来后,我们自己实现的shell可以实现一些基本指令,如:ls ls-l pwd touch rm 等