进程替换以及简单实现简易shell

       在前面学习了进程的创建、等待、退出之后,我们知道了在有些时候子进程和父进程会共享地址空间,比如在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:


 好了,对于进程今天我们就先学到这。多有不足,望大佬们多指点。

猜你喜欢

转载自blog.csdn.net/weixin_38175489/article/details/79745395