fork

pid_t fork(void);
  1. 它在调用进程(成为父进程)中返回一次,返回值为新派生进程(成为子进程)的进程ID号
  2. 在子进程中又返回一次,返回值为0.因此,返回值本身告知当前进程是子进程还是父进程

 fork在子进程中返回0而不是父进程的ID的原因:

  1. 任何子进程只有一个父进程,而且子进程总是可以通过调用getppid取得父进程的ID
  2. 父进程可以有许多子进程,而且无法获得各个子进程的进程ID。如果父进程想要跟踪所有子进程的ID,那么它必须记录每次调用fork的返回值。

  父进程中调用fork之前打开所有的描述字在fork返回之后由子进程分享。父进程调用accept之后调用fork。所接受的已连接的套接口随后就在父进程与子进程之间分享。通常情况下,子进程接着读和写这个套接口,父进程则关闭这个已连接套接口。
用法:

  1. 一个进程创建一个自身的拷贝,这样每个拷贝都可以在另一个拷贝执行其他任务的同时处理各自的某个操作。这是网络服务器的典型用法。
  2. 一个进程想要执行另一个程序。既然创建新进程的唯一方法为调用fork,该进程于是首先调用fork创建一个自身的拷贝,然后其中一个拷贝(通常为子进程)调用exec把自身替换成新的程序。这是诸如shell之类程序的典型用法。
  3. fork产生的子进程将继承父进程的信号掩码,但具有一个空的挂起信号集

exec

  存放在硬盘上的可执行文件能够被UNIX执行的唯一方法是:由一个现有进程调用六个exec函数中的某一个

  1. exec把当前进程映像替换成新的进程文件,而且该新程序通常从main函数处开始执行
  2. 进程ID并不改变
  3. 我们称调用exec的进程为调用进程,称新执行的程序为新程序
#include <unistd.h>

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[]);
int execve(const char *path, char *const argv[], char *const envp[]);

六个exec函数的区别

  1. 待执行的程序文件是由文件名还是由路径名指定
  2. 新程序的参数是一一列出还是由一个指针数组来引用
  3. 把调用进程的环境传递给新程序还是给新程序指定新的环境

  这些函数只在出错时才返回到调用者。否则,控制将传递给新程序的起始点,通常就是main函数。
这六个函数之间的关系如下图所示。一般来说,只有 execve 是内核中的系统调用,其他五个都是调用 execve 的库函数。

  1. 顶行三个函数把新程序的每个参数字符串指定成exec的一个独立参数,并以一个空指针结束可变数量的这些参数。底行三个函数都有一个作为exec参数的argv数组,其中含有指向新程序各个参数字符串的所有指针。既然没有指定参数字符串的数目,这个argv数组必须含有一个用于指定其末尾的空指针。
  2. 左列两个函数指定一个filename参数。exec将使用当前的PATH环境变量把该文件名参数转换为一个路径名。然而如果这两个函数的filename参数中不论何处含有一个斜杠(/),PAHT变量就不再使用。右两列四个函数指定一个全限定的pathname参数。
  3. 左两列四个函数不显式指定一个环境指针。相反,他们使用外部变量environ的当前值来构造一个传递给新程序的环境清单。右列两个函数显式指定一个环境清单,其envp指针数组必须以一个空指针结束。
    //execv
        int childpid;
        int i;
    
        )
        {
            //child process
            char * execv_str[] = {"echo", "executed by execv",NULL};
             )
            {
                perror("error on exec");
                exit();
            }
        }
        else
        {
            //parent process
            wait(&childpid);
            printf("execv done\n\n");
        }
    
    //execvp
        )
        {
            //child process
            char * execvp_str[] = {"echo", "executed by execvp",">>", "~/abc.txt",NULL};
             )
            {
                perror("error on exec");
                exit();
            }
        }
        else
        {
            //parent process
            wait(&childpid);
            printf("execvp done\n\n");
        }
    
    //execve
        )
        {
            //child process
            char * execve_str[] = {"env",NULL};
            char * env[] = {"PATH=/tmp", "USER=lei", "STATUS=testing", NULL};
             )
            {
                perror("error on exec");
                exit();
            }
        }
        else
        {
            //parent process
            wait(&childpid);
            printf("execve done\n\n");
        }
    
    //execl
        )
        {
            //child process
             )
            {
                perror("error on exec");
                exit();
            }
        }
        else
        {
            //parent process
            wait(&childpid);
            printf("execv done\n\n");
        }
    //execlp
        )
        {
            //child process
             )
            {
                perror("error on exec");
                exit();
            }
        }
        else
        {
            //parent process
            wait(&childpid);
            printf("execv done\n\n");
        }
    
    //execle
        )
        {
            //child process
            char * env[] = {"PATH=/home/gateman", "USER=lei", "STATUS=testing", NULL};
            ){
                perror("error on exec");
                exit();
            }
        }
        else
        {
            //parent process
            wait(&childpid);
            printf("execle done\n\n");
        }

fork和exec的更多相关文章

  1. 【转】Linux下Fork与Exec使用

    Linux下Fork与Exec使用 转自 Linux下Fork与Exec使用 一.引言 对于没有接触过Unix/Linux操作系统的人来说,fork是最难理解的概念之一:它执行一次却返回两个值.for ...

  2. Linux下Fork与Exec使用

    Linux下Fork与Exec使用   一.引言 对于没有接触过Unix/Linux操作系统的人来说,fork是最难理解的概念之一:它执行一次却返回两个值.fork函数是Unix系统最杰出的成就之一, ...

  3. linux进程之fork 和 exec函数

    ---恢复内容开始--- fork函数 该函数是unix中派生新进程的唯一方法. #include <unistd.h> pid_t   fork(void); 返回: (调用它一次, 它 ...

  4. linux c语言 fork() 和 exec 函数的简介和用法

    linux c语言 fork() 和 exec 函数的简介和用法   假如我们在编写1个c程序时想调用1个shell脚本或者执行1段 bash shell命令, 应该如何实现呢? 其实在<std ...

  5. Linux下Fork与Exec

    一.引言 对于没有接触过Unix/Linux操作系统的人来说,fork是最难理解的概念之一:它执行一次却返回两个值.fork函数是Unix系统最杰出的成就之一,它是七十年代UNIX早期的开发者经过长期 ...

  6. fork 和 exec

    https://blog.csdn.net/disadministrator/article/details/39347333 进程创建方法:fork.exec.clone,父进程等待子进程结束是用w ...

  7. fork、exec 和 exit 对 IPC 对象的影响

    GitHub: https://github.com/storagezhang Emai: debugzhang@163.com 华为云社区: https://bbs.huaweicloud.com/ ...

  8. Linux fork()、exec() Hook Risk、Design-Principle In Multi-Threadeed Program

    目录 . Linux exec指令执行监控Hook方案 . 在"Multi-Threadeed Program"环境中调用fork存在的风险 . Fork When Multi-T ...

  9. fork和exec一起使用

    先预览一下工程的目录树: 实现的功能:master进程启动slave进程. 看看Makefile内容: all: master.out slave.out master.out: master.cpp ...

  10. fork和exec函数

    #include<unistd.h> pid_t fork(void); 返回:在子进程中为0,在父进程中为子进程IO,若出错则为- fork最困难之处在于调用它一次,它却返回两次.它在调 ...

随机推荐

  1. DevExpress WPF v18.2新版亮点(一)

    买 DevExpress Universal Subscription  免费赠 万元汉化资源包1套! 限量15套!先到先得,送完即止!立即抢购>> 行业领先的.NET界面控件2018年第 ...

  2. 201621123001 《Java程序设计》第14周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结与数据库相关内容. Mysql数据库简单操作,常用的操作命令 启动:进入Mysql (从命令行mysql -u root -p) 退 ...

  3. findbugs 安装及使用

    1.eclipse安装findbugs插件 Help -> install new softWare 输入 http://findbugs.cs.umd.edu/eclipse 2.使用find ...

  4. system的共享内存实例

    system的共享内存指的是内核指定一块内存区域映射到虚拟地址空间供进程通信使用的机制 1\创建或打开共享内存块函数原型int shmget(key_t key, size_t size, int s ...

  5. 强化学习10-Deep Q Learning-fix target

    针对 Deep Q Learning 可能无法收敛的问题,这里提出了一种  fix target 的方法,就是冻结现实神经网络,延时更新参数. 这个方法的初衷是这样的: 1. 之前我们每个(批)记忆都 ...

  6. Java语法基础学习DaySix

    一.JavaBean——可重用组件 1.JavaBean是指符合以下标准的Java类: (1)类是公共的 (2)有一个无参的公共的构造器 (3)有属性,且有对应的get.set方法 2.好处 用户可以 ...

  7. SparkStreaming实时日志分析--实时热搜词

    Overview 整个项目的整体架构如下: 关于SparkStreaming的部分: Flume传数据到SparkStreaming:为了简单使用的是push-based的方式.这种方式可能会丢失数据 ...

  8. L299 EST 科技英语翻译-美学取向 (下)

    4. Ordering(有序美) DescriptiveExpositoryArgumentative Chinese: end focus 句尾焦点English: beginning focus ...

  9. mysql 数据库关于my.int 的相关问题

    最好在建库的时候直接建好 create database db1 charset utf8; my.int  在mysql的目录里 名曰配置文件    里面主要是内容就是 1 一般用到的就是编码不统一 ...

  10. Python 再次改进版通过队列实现一个生产者消费者模型

    import time from multiprocessing import Process,Queue #生产者 def producer(q): for i in range(10): time ...