zoukankan      html  css  js  c++  java
  • Linux-进程虚拟地址空间中加载新映像(续进程基础)

    5、在进程虚拟地址空间中加载新映像

      在子进程的虚拟地址空间里加载新的映像,需要使用系统提供的一系列函数:

      

      他们的作用都是执行一个文件,当我们创建了一个进程之后,通常将子进程替换成新的进程映象,这可以用exec系列的函数来进行。当然,exec系列的函数也可以将当前进程替换掉。例如:在shell命令行执行ps命令,实际上是shell进程调用fork复制一个新的子进程,在利用exec系统调用将新产生的子进程完全替换成ps进程。

    exec系列函数(execl、execlp、execle、execv、execvp)包含头文件<unistd.h>
    
    功能:
    
        用exec函数可以把当前进程替换为一个新进程,且新进程与原进程有相同的PID。exec名下是由多个关联函数组成的一个完整系列,
    
    头文件<unistd.h>
    
    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 execvp(const char *file, char *const argv[]);

      exec后面字母的含义:

      l list

      v vector

      p PATH

      e 环境变量

    注:上述exec系列函数底层都是通过execve系统调用实现:
    
    #include <unistd.h>
    
    int execve(const char *filename, char *const argv[],char *const envp[]);
    
    DESCRIPTION: 
           execve() executes the program pointed to by filename.  filename must be 
           either a binary executable, or a script starting with  a  line  of  the form 

    返回值:  
      错误:-1
      成功:不返回,errno被设置

    例:使用如下命令选项


        

    复制代码
    以上exec系列函数区别: 1,带l 的exec函数:execl,execlp,execle,表示后边的参数以可变参数的形式给出且都以一个空指针结束。 示例:
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    
    int main(void)
    {
        printf("entering main process---
    ");
        execl("/bin/ls","ls","-l",NULL);
        printf("exiting main process ----
    ");
        return 0;
    }

    利用execl将当前进程main替换掉,所有最后那条打印语句不会输出
    复制代码
    
    
    复制代码
    2,带 p 的exec函数:execlp,execvp,表示第一个参数path不用输入完整路径,只有给出命令名即可,它会在环境变量PATH当中查找命令
    示例:
    当不带p但没给出完整路径时:
    #include <stdio.h>
    #include <stdlib.h>
    #include <unistd.h>
    int main(void)
    {
        printf("entering main process---
    ");
        if(execl("ls","ls","-l",NULL)<0)
          perror("excl error");
        return 0;
    }

    结果显示找不到,所有替换不成功,main进程继续执行

    现在带p:

    if(execlp("ls","ls","-l",NULL)<0)

    替换成功
    复制代码
    复制代码
    3,不带 l 的exec函数:execv,execvp表示命令所需的参数以char *arg[]形式给出且arg最后一个元素必须
    
    是NULL
    
    示例:
    
    #include <stdio.h> #include <stdlib.h> #include <unistd.h> int main(void) { printf("entering main process---
    "); int ret; char *argv[] = {"ls","-l",NULL}; ret = execvp("ls",argv); if(ret == -1) perror("execl error"); printf("exiting main process ----
    "); return 0; }
    替换成功

    复制代码
    4,带 e 的exec函数:execle表示,将环境变量传递给需要替换的进程
    
    从上述的函数原型中我们发现:
    
    extern char **environ;
    
    此处的environ是一个指针数组,它当中的每一个指针指向的char为“XXX=XXX”
    
    environ保存环境信息的数据可以env命令查看:

    关于bash下敲命令的执行原理:

    (1)bash的内部命令和外部命令

      bash下的命令分内部命令和外部命令,外部命令可以使用which查看,而内部命令不能被查看,外部命令执行需要重新fork,并将外部命令的可执行程序加载的fork新建的进程虚拟地址空间。而内部命令的调用(内部命令是bash的一部分),其实就是调用内部的函数。

    (2)外部命令执行

      当在bash下输入如下命令时,

        

    "ps -o  pid,ppid,pgrp,comn"

      bash先进行fork;解析命令行参数为char *const ps_argv={"ps","-o","pid,ppid,pgrp,comn"}

    再在新进程的虚拟地址空间上使用execvp函数加载ps 可执行程序,并ps_argv作为参数。

    (3)内部命令执行

      调用内部函数,不新建进程。

    (4)如何查看一个命令是内部命令还是外部命令,使用type查看。type+需要查看的命令

    使用system(3)启动新的可执行程序
    #include<stdlib.h>
    int system(const char *command);
    功能:
      执行一个shell命令
    参数:
      command:可执行命令
    返回值:
      错误:-1
      成功:返回command的退出状态码
    在子进程中加载ls命令:
    pid_t pid;
    pid=fork();
    if(pid==-1){
         return 1;
    }
    if(pid==0){
         system("ls -l");
         exit(0);   
    }
    else{
         wait(NULL);
    }


    system函数和exec系列函数的区别
    pid_t pid=fork();
    if(pid==-1){
         perror("fork");
         return 1;  
    }
    if(pid==0){
         //加载新的映像
         
       system("myt");//myt为等待字符输入的映像 }
    else{ wait(NULL); }

      

    pid_t pid=fork();
    if(pid==-1){
         perror("fork");
         return 1;  
    }
    if(pid==0){
         //加载新的映像
         
       execl("./myt","myt","NULL");//myt为等待字符输入的映像
    
    }
    else{
         wait(NULL);    
    }

      如下图,在bash下启动a.out,a.out在子进程被执行,子进程启动了shell,又在shell下启动了myt,即system通过shell来解析了myt。bash——a.out——a.out——shell——myt,相当于执行/bin/sh -c command。;而exec系列函数是bash先进行fork,解析命令行参数,

    再在新进程的虚拟地址空间上使用execvp函数加载命令行可执行程序,并解析的命令作为参数,此时的进程树:bash——a.out——myt。

    system函数下,查看进程树:

    excel函数查看进程树:

  • 相关阅读:
    九度oj 题目1465:最简真分数
    九度oj 题目1083:特殊乘法 清华大学2010年机试题目
    九度oj 题目1084:整数拆分 清华大学2010年机试题目
    九度oj 题目1085:求root(N, k) 清华2010年机试题目
    九度oj 题目1460:Oil Deposit
    九度oj 题目1459:Prime ring problem
    九度oj 题目1458:汉诺塔III
    九度oj 题目1457:非常可乐
    题目1451:不容易系列之一
    移动端滚动不流畅,添加-webkit-overflow-scrolling属性 值为touch
  • 原文地址:https://www.cnblogs.com/ptfe/p/11001018.html
Copyright © 2011-2022 走看看