1.编写一个伪终端:在真正的终端上运行这个伪终端程序之后,能够执行所有的shell指令,甚至再次运行自己
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <dirent.h>
#include <wait.h>
/*
编写一个伪终端:在真正的终端上运行这个伪终端程序之后,能够执行所有的shell指令,甚至再次运行自己
*/
/*
char *strtok(char *str,const char *delim)
{
static char buf[n]={0};
if(str==NULL)
{
分割buf
}
else
{
分割str
strcpy(buf,分割完的剩下的str);
}
}
char* res=strtok("hello world 你好中国"," ")
res =="hello"
static_buf =="world 你好中国"
char*res =strtok(NULL," ");
char str[128]="hello world 你好中国"
*/
/*
\033[显示方式;前景色;背景色m
*/
void getstring(char* buf,int size)
{
fgets(buf,size,stdin);
int len=strlen(buf);
if(buf[len-1]==10){
buf[len-1]=0;
}
}
int main(int argc, char *argv[])
{
while(1)
{
int ret=fork();
if(ret>0)
{
//在循环fork的只有父进程
char* username=getlogin();
char hostname[128]={0};
char path[128]={0};
gethostname(hostname,128);
getcwd(path,128);
printf("\033[1;32;10m%s@%s\033[0m:\033[1;34;10m%s\033[0m$",username,hostname,path);
fflush(stdout);
wait(0);
}
else if(ret == 0)
{
// 因为子进程会执行进程替换, 所以根本不担心子进程会循环回来继续fork
char cmd[256] = {0};
getstring(cmd,256);// cp 111. c 222. c
// 准备一个传递给execvp的参数的字符串数组
char* arg[32] = {0};
int i = 0;// 用作arg下标
char* res = NULL;//用来接受strtok的返回值
do
{
if(res == NULL)
{
res = strtok(cmd," ");//分割cmd
}
else
{
res = strtok(NULL," ");//分割cmd里面剩下的字符串
}
if(res == NULL){break;}
arg[i++] = res;//将分割出来的结果存放到arg数组里面, 并且下标自增1
}while(1);
// 由于cd不是shell指令, 所以我们要做判断, 如果arg[0]是cd的话, 则不执行进程替换, 而是执行chdir
if(strcmp(arg[0],"cd")==0)
{
if(arg[1][0]==0 || strcmp(arg[1],"~")==0)
{
chdir("/home/gyf123q");
}
else
{
chdir(arg[1]);
}
}
else
{
execvp(arg[0], arg);
}
}
else
{
}
}
return 0;
}