操作系统实验二归纳

2020-07-14 14:17:17 浏览数 (1)

本实验是要求在linux环境下测试fork()和exec(),并建立一个简单的shell(带cd、env、echo、help、jobs、quit命令)

  • fork()

这一节没啥难度,主要是测试当前linux环境下gcc是否能编译成功

代码语言:javascript复制
清单 2-1 创建进程
#include <stdio.h>   //此处指导书上没有
int main (){
    int x;
    while((x=fork())==-1);
    if (x==0)
        printf("a");
    else
        printf("b");
printf("c");
}

这里出现了很奇妙的问题,指导书上没有使用stdio头文件,不过显然这样编译不会通过。但是加上了以后又出现了一个坑,实验室里的linux没有安装gcc的包依赖。此处需要我们重装个gcc

代码语言:javascript复制
$ sudo apt-get install build-essential  //此处也许会失败,提示缺少几个依赖,那就安装相应的包
$ sudo apt-get install g     //上一步如果不行就运行这个再运行上一条

linux下怎么build .c文件也留档一下吧 $ gcc -o hello hello.c // 第一个为生成的文件名,第二个为.c文件

  • exec()

这真是一个骚操作。可以直接用新的进程映象置换当前的进程映象,留档函数原型

exec 系列有 6 个函数,原型如下: extern char *environ; 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 execve (const char filename, char const argv [], char const envp[]); int execvp( const char file, char const argv[]); exec 系列函数用新的进程映象置换当前的进程映象.这些函数的第一个参数是待执行程序的路 径名(文件名)。这些函数调用成功后不会返回,其进程的正文(text),数据(data)和栈(stack)段被待执行 程序程序覆盖。但是进程的 PID 和所有打开的文件描述符没有改变,同时悬挂信号被清除,信号重 置为缺省行为。 在函数 execl,execlp,和 execle 中, const char arg 以及省略号代表的参数可被视为 arg0, arg1, …,argn。它们合起来描述了指向 NULL 结尾的字符串的指针列表,即执行程序的参数列表。作 为约定,第一个 arg 参数应该指向执行程序名自身,参数列表必须用 NULL 指针结束。 execv 和 execvp 函数提供指向 NULL 结尾的字符串的指针数组作为新程序的参数列表。作为约 定,指针数组中第一个元素应该指向执行程序名自身。指针数组必须用 NULL 指针结束。 execle 函数同时说明了执行进程的环境(environment),它在 NULL 指针后面要求一个附加参 数,NULL 指针用于结束参数列表,或者说,argv 数组。这个附加参数是指向 NULL 结尾的字符串的指 针数组,它必须用 NULL 指针结束。其它函数从当前进程的 environ 外部变量中获取新进程的环境。 execlp和execvp可根据path搜索合适的程序运行,其它则需要给出程序全路径。 execve()类似 execv(),但是加上了环境的处理。

在把此处的示例程序解释一下吧:

代码语言:javascript复制
#include <sys/types.h>
#include <stdio.h>
#include <unistd.h>
int main(){
        pid_t pid;
        /* fork a child process */
        pid = fork();
        if (pid < 0){
        /* error occurred */
                fprintf(stderr, "Fork Failed");
                return 1;
        }
        else if (pid == 0){ 
        /* 子进程 */
        execlp("/bin/ls","ls",NULL);
        }
        else { /* 父进程 */
        / * 父进程将一直等待,直到子进程运行完毕*/
        wait(NULL);
        printf("Child Complete");
        }
return 0;
}

我的理解是这份代码的意思应该是,创建一个进程,如果该进程的pid<0(当然创建fork成功会返回大于0的数)提示fork失败。然后再使用execlp调用系统中的ls命令,运行结果图

  • ##简单的shell
代码语言:javascript复制
#include <sys/types.h>
#include <sys/wait.h>

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>

char op[256];
char arg[256];
char str[1024];

int main()
{
    pid_t CurSon;

    while(1){
        scanf("%s", op);
        if(strcmp(op, "quit") == 0 || strcmp(op, "exit") == 0 || strcmp(op, "bye") == 0){
            printf("nQuiting...Bye~");
            return 0;
        }else if(strcmp(op, "help") == 0){
            printf("It's a fake shell~n");
        }else if(strcmp(op, "cd") == 0){
            scanf("%s", arg);
            sprintf(str, "cd %s", arg);
            system(str);
        }else if(strcmp(op, "environ") == 0){
            system("env");
        }else if(strcmp(op, "echo") == 0){
            scanf("%s", arg);
            sprintf(str, "echo %s", arg);
            system(str);
        }else if(strcmp(op, "jobs") == 0){
            system("ps");
        }else{
            fgets(arg, 256, stdin);
            sprintf(str, "%s %s", op, arg);
            while((CurSon = fork()) < 0);
            if(CurSon == 0){
                system(str);
                return 0;
            }else{
                wait(NULL);
                printf("%sn", str);
                printf("Complete...n");
            }
        }
    }
    return 0;
}

0 人点赞