zoukankan      html  css  js  c++  java
  • 2017-2018-1 20155334 《信息安全系统设计基础》第六周学习总结

    2017-2018-1 20155334 《信息安全系统设计基础》第六周学习总结

    1. fork() 函数:

    1. 一个进程,包括代码、数据和分配给进程的资源。
    2.  fork() 函数通过系统调用创建一个与原来进程几乎完全相同的进程,也就是两个进程可以做完全相同的事,但如果初始参数或者传入的变量不同,两个进程也可以做不同的事。
    3. 一个进程调用fork() 函数后,系统先给新的进程分配资源,例如存储数据和代码的空间。然后把原来的进程的所有值都复制到新的新进程中,只有少数值与原来的进程的值不同。相当于克隆了一个自己。
    
    在fork函数执行完毕后,如果创建新进程成功,则出现两个进程,一个是子进程,一个是父进程。
    fork调用的一个奇妙之处就是它仅仅被调用一次,却能够返回两次,它可能有三种不同的返回值:
    1)在父进程中,fork返回新创建子进程的进程ID;
    2)在子进程中,fork返回0;
    3)如果出现错误,fork返回一个负值;
    
    我们可以通过fork返回的值来判断当前进程是子进程还是父进程
    

    有如下代码:

    #include <unistd.h>  
    #include <stdio.h>  
    int main(void)  
    {  
       int i=0;  
       printf("i son/pa ppid pid  fpid/n");  
       //ppid指当前进程的父进程pid  
       //pid指当前进程的pid,  
       //fpid指fork返回给当前进程的值  
       for(i=0;i<2;i++){  
           pid_t fpid=fork();  
           if(fpid==0)  
               printf("%d child  %4d %4d %4d/n",i,getppid(),getpid(),fpid);  
           else  
               printf("%d parent %4d %4d %4d/n",i,getppid(),getpid(),fpid);  
       }  
       return 0;  
    }  
    

    运行结果如下:

    分析:
    第一次fork后,p3224(父进程)的变量为i=0,fpid=3225(fork函数在父进程中返向子进程id)p3225(子进程)的变量为i=0,fpid=0(fork函数在子进程中返回0)
    第二步创建了两个进程p3226,p3227,这两个进程执行完printf函数后就结束了,因为这两个进程无法进入第三次循环,无法fork,该执行return 0;了,其他进程也是如此。

    2. exec函数:

    1. fork函数是用于创建一个子进程,该子进程几乎是父进程的副本,而有时我们希望子进程去执行另外的程序,exec函数族就提供了一个在进程中启动另一个程序执行的方法。
    2. 它可以根据指定的文件名或目录名找到可执行文件,并用它来取代原调用进程的数据段、代码段和堆栈段,在执行完之后,原调用进程的内容除了进程号外,其他全部被新程序的内容替换了。
    3. 这里的可执行文件既可以是二进制文件,也可以是Linux下任何可执行脚本文件。
    

    exec函数族使用注意点:
    在使用exec函数族时,一定要加上错误判断语句。因为exec很容易执行失败,其中最常见的原因有:
    ① 找不到文件或路径,此时errno被设置为ENOENT。
    ② 数组argv和envp忘记用NULL结束,此时errno被设置为EFAULT。
    ③ 没有对应可执行文件的运行权限,此时errno被设置为EACCES。

    在Linux中使用exec函数族主要有以下两种情况

    如果一个进程想执行另一个程序,那么它就可以调用fork函数新建一个进程,然后调用任何一个exec函数使子进程重生。
    当进程认为自己不能再为系统和用户做出任何贡献时,就可以调用任何exec 函数族让自己重生。
    

    execlp.c文件如下:

    #include <stdio.h>
    #include <unistd.h>
    int main()
    {
        if(fork()==0){
            if(execlp("/usr/bin/env","env",NULL)<0)
            {
                perror("execlp error!");
                return -1 ;
            }
        }
        return 0 ;
    }
    

    执行结果如图:

    由执行结果看出,execlp函数使执行码重生时继承了Shell进程的所有环境变量,其他三个不以e结尾的函数同理。

    3. wait()函数:

    1. wait()会暂时停止目前进程的执行, 直到有信号来到或子进程结束. 
    2. 如果在调用wait()时子进程已经结束, 则wait()会立即返回子进程结束状态值. 
    3. 子进程的结束状态值会由参数status 返回, 而子进程的进程识别码也会一快返回
    

    4使用fork,exec,wait实现mybash:

    部分代码展示,全部代码在码云里

    int main()
    {
        char cmdline[MAX];
        
        while(1){
            printf("bsetixx@besrixx-VirtualBox:~/XINAN/mybash/$ ");
            fgets(cmdline,MAX,stdin);
            if(feof(stdin))
            {
                printf("error");
                exit(0);
            }
            eval(cmdline);
        }
    }
    
    void eval(char *cmdline)
    {
        char *argv[MAX];
        char buf[MAX];
        int bg;
        pid_t pid;
        strcpy(buf,cmdline);
        bg = parseline(buf,argv);
        if(argv[0]==NULL)
            return;
        if(!builtin_command(argv)) 
        {   
        if((pid=fork()) == 0)
        {
            if(execvp(argv[0],argv) < 0) {
                printf("%s : Command not found.
    ",argv[0]);
                exit(0);
            }
        }
    
        if(!bg){
            int status;
            if(waitpid(-1,&status,0) < 0) 
                printf("waitfg: waitpid error!");
        }
        else
            printf("%d %s",pid, cmdline);
        return;
        }
    }
    
    

    第十章 教材学习内容总结

    1.打开和关闭文件
    (1)是通过调用open函数来创建一个新文件或者打开一个已存在的文件:

    include <sys/types.h>
    
    include <sys/stat.h>
    
    

    (2)创建一个新文件,要使得文件的拥有者有读写权限,但是所有其他的用户都有只读权限

    umask(DEF_UMASK);
    fg=Open("foo.txt",O_CREAT|O_TRUNC|O_WEONLY,DEF_MODE);
    

    (3)通过调用close函数关闭一个打开的文件

    int close(int fd);
    

    返回值成功为0,出错为-1

    2.读和写文件

    ①读时遇到EOF。假设我们读一个文件,该文件从当前文件位置开始只含有20多个字节,而我们以50个字节的片进行读取。这样一来,下一个read返回的不足值为20,此后的read将通过返回不足值0来发出EOF信号。

    ②从终端读文本行。如果打开文件是与终端相关联的(如键盘和显示器),那么每个read函数将以此传送一个文本行,返回的不足值等于文本行的大小。

    ③读和写网络套接字。如果打开的文件对应于网络套接字,那么内部缓冲约束和较长的网络延迟会引起read和write返回不足值。对Unix管道调用read和write时,也有可能出现不足值,这种进程间的通信机制不在我们讨论的范围之内。

    实际上,除了EOF,在读磁盘文件时,将不会遇到不足值,而且在写磁盘文件时,也不会遇到不足值。如果想创建简装的诸如web服务器这样的网络应用,就必须通过反复调用read和write处理不足值,直到所有需要的字节都传送完毕。

    3.读取文件元数据

    (1)应用程序能够通过调用stat和fstat函数,检索到关于文件的信息(元数据)

    (2)文件类型

    ①目录文件:包含其他文件的信息,宏指令:S_ISDIR()

    ②普通文件:二进制或文本数据,宏指令:S_ISREG()

    ③套接字:通过网络和其他进程通信的文件,宏指令:S_ISSOCK()

    4.I/0重定向

    (1)重定向使用dup2函数
    (2)Unix外壳提供了I/O重定向操作符,允许用户将磁盘文件和标准输入输出联系起来

    代码提交截图

    代码提交连接

  • 相关阅读:
    牛客多校(2020第三场)C Operation Love
    牛客多校(2020第三场)C Operation Love
    牛客多校(2020第三场)B Classical String Problem
    牛客多校(2020第三场)B Classical String Problem
    牛客多校(2020第三场)L Problem L is the Only Lovely Problem
    牛客多校(2020第三场)L Problem L is the Only Lovely Problem
    一个图形或者控件旋转时 判断方向逆时针还是顺时针
    为什么 dll 改名字之后无法使用
    C# 几个特殊运算符的理解和Nullable<T> 的研究
    再次深入 C# Attribute
  • 原文地址:https://www.cnblogs.com/bestixx/p/7750646.html
Copyright © 2011-2022 走看看