一:fork()和vfork()的区别:

   fork()函数可以创建子进程,有两个返回值,即调用一次返回两个值,一个是父进程调用fork()后的返回值,该返回值是刚刚创建的子进程的ID;另一个是子进程调用fork()后的返回值,该返回值为0。

   vfork与fork不同的地方在于:

   使用fork()创建子进程时:子进程只是完全复制父进程的资源,并且哪个进程先运行取决于系统的调度算法。

点击(此处)折叠或打开

  1. int globVar = 5;
  2. int main(int argc,char *argv[])
  3. {
  4. int var = 1,i;
  5. pid_t pid;
  6. printf("fork is different with vfork\n");
  7. pid = fork();
  8. // pid = vfork();
  9. printf("%d---------\n",pid);
  10. switch(pid)
  11. {
  12. case 0:
  13. i = 2;
  14. while(i-- > 0)
  15. {
  16. printf("--------Child process is running\n");
  17. globVar++;
  18. var++;
  19. printf("--------c sleep\n");
  20. sleep(1);
  21. }
  22. printf("--------Clild's globVar = %d,var = %d\n",globVar,var);
  23. break;
  24. case -1:
  25. perror("Process creat faild\n");
  26. exit(0);
  27. default:
  28. i = 3;
  29. while(i-- > 0)
  30. {
  31. printf("Parent process is running\n");
  32. globVar++;
  33. var++;
  34. printf("p sleep\n");
  35. sleep(1);
  36. }
  37. printf("Parent's globVar = %d,var = %d\n",globVar,var);
  38. exit(0);
  39. }
  40. }
  41. fork is different with vfork
  42. 14070---------
  43. Parent process is running
  44. p sleep
  45. 0---------
  46. --------Child process is running
  47. --------c sleep
  48. Parent process is running
  49. p sleep
  50. --------Child process is running
  51. --------c sleep
  52. Parent process is running
  53. p sleep
  54. --------Clild's globVar = 7,var = 3
  55. Parent's globVar = 8,var = 4

使用vfork()创建子进程时:操作系统并不将父进程的地址空间完全复制到子进程,子进程共享父进程的地址空间,并且保证子进程先运行。

点击(此处)折叠或打开

  1. 将上面的例子vfork()注释去掉,而将fork();注释掉,可以得到与之不同的结果
  2. fork is different with vfork
  3. 0---------
  4. --------Child process is running
  5. --------c sleep
  6. --------Child process is running
  7. --------c sleep
  8. --------Clild's globVar = 7,var = 3
  9. 14160---------
  10. Parent process is running
  11. p sleep
  12. Parent process is running
  13. p sleep
  14. Parent process is running
  15. p sleep
  16. Parent's globVar = 10,var = 32714

二:signal函数,可以用来处理我们系统的一些信号。

typedef void (*sighandler_t)(int);

sighandler_t signal(int signum, sighandler_t handler));

函数的第一个参数指定该信号的值,第二个参数指定对该信号的处理。可以忽略该信号。参数设为(SIG_ING),也可以采用系统默认的方式,参数设置为(SIG_DFL),下面的例子就是忽略我们的输入信号。

点击(此处)折叠或打开

  1. #include<stdio.h>
  2. #include<signal.h>
  3. int main(int argc ,char *argv[])
  4. {
  5. signal(SIGINT,SIG_IGN);
  6. for(;;);
  7. return 0;
  8. }

执行结果就是无限循环,我们ctrl+c也无法停止。

三:exec系列函数(execl,execlp,execle,execv,execvp)函数用法总结:

   exec系列函数均可以把当前进程替换成一个新进程,保持原有的pid不变。对于以上函数,我按照自己的理解分为三类,当然这不一定是最好的方法,具体看自己了。

   1:execl和execlp:我们给它传入相应命令和参数,它就可以脱离当前进程而独立出来。他们两个的不同在于第一个参数传绝对路径还是相对路径,请看下面例子:

点击(此处)折叠或打开

  1. #include<stdio.h>
  2. #include<stdlib.h>
  3. #include<unistd.h>
  4. int main(int argc,char *argv[])
  5. {
  6. printf("entering main process----\n");
  7. // execl("/bin/ls","ls","-l","/home",NULL);
  8. // execlp("ls","ls","-l","/home",NULL);
  9. printf("exiting main process -----\n");
  10. return 0;
  11. }

程序的执行结果就是我/home中的内容了,而第10行的内容并不会被打印出来。有没有懂一点点呢,继续;

   2:execv和execvp函数:同样,后面有p就是它可以只传入文件名或者命令名就可以执行,不过它的参数可以自己用argv[]传进去。

点击(此处)折叠或打开

  1. int ret;
  2. printf("entering main process----\n");
  3. char *argv[] = {"ls","-l","/home",NULL};
  4. // ret = execv("/bin/ls",argv);
  5. ret = execvp("ls",argv);
  6. if(ret == -1)
  7. {
  8. perror("execvp error");
  9. }
  10. printf("exiting main process-----\n");
  11. return 0;

看吧,主要是execv和execvp只能传入两个参数,所以把其他的东西就放在argv[]中了,执行成功返回0,失败返回-1。

   3:execle函数,可以获得环境变量,看下面例子:

(注意:test_execle_child 是一个可以打印当前环境变量的函数)

test_execle_child函数如下:

点击(此处)折叠或打开

  1. extern char ** environ;
  2. int main(int argc,char *argv[])
  3. {
  4. int i ;
  5. printf("\n\n现在已经进入main函数中的子程序---- pid=%d\n",getpid());
  6. sleep(2);
  7. for(i = 0;environ[i] != NULL;i++)
  8. {
  9. printf("%s\n",environ[i]);
  10. }
  11. return 0;
  12. }

点击(此处)折叠或打开

  1. int ret;
  2. char * const envp[] = {"aa=11","bb=22","cc=33",NULL};
  3. printf("entering main process----\n");
  4. // ret = execl("./test_execle_child","test_execle_child",NULL);
  5. ret = execle("./test_execle_child","test_execle_child",NULL,envp);
  6. if(ret == -1)
  7. {
  8. perror("execle error");
  9. }
  10. printf("exiting main process-----\n");
  11. return 0;

当函数是execle时:输出我们envp中传进去的环境变量,

当函数时execl时:输出我们当前执行进程的环境变量。

阅读(1) | 评论(0) | 转发(0) |

给主人留下些什么吧!~~
评论热议

版权声明:本文为博主原创文章,未经博主允许不得转载。

fork()和vfork()的区别,signal函数用法,exec()系列函数的用法小结的更多相关文章

  1. fork()和vfork()的区别(转载)

    fork和vfork 转载 http://coolshell.cn/articles/12103.html 在知乎上,有个人问了这样的一个问题--为什么vfork的子进程里用return,整个程序会挂 ...

  2. exec系列函数(execl,execlp,execle,execv,execvp)使用

    本节目标: exec替换进程映像 exec关联函数组(execl.execlp.execle.execv.execvp) 一,exec替换进程映像 在进程的创建上Unix采用了一个独特的方法,它将进程 ...

  3. linux系统编程之进程(五):exec系列函数(execl,execlp,execle,execv,execvp)使用

    本节目标: exec替换进程映像 exec关联函数组(execl.execlp.execle.execv.execvp) 一,exec替换进程映像 在进程的创建上Unix采用了一个独特的方法,它将进程 ...

  4. exec系列函数和system函数

    一.exec替换进程映象 在进程的创建上Unix采用了一个独特的方法,它将进程创建与加载一个新进程映象分离.这样的好处是有更多的余地对两种操作进行管理.当我们创建 了一个进程之后,通常将子进程替换成新 ...

  5. fork和vfork的区别

    参见百度百科API说明: fork 头文件: #include<unistd.h> #include<sys/types.h> 函数原型: pid_t fork( void); ...

  6. Linux系统编程(9)—— 进程之进程控制函数exec系列函数

    在Linux中,并不存在exec()函数,exec指的是一组函数,一共有6个,分别是: #include <unistd.h> extern char **environ; int exe ...

  7. 进程基本-进程创建,僵尸进程,exec系列函数

    Linux系统中,进程的执行模式划分为用户模式和内核模式,当进程运行于用户空间时属于用户模式,如果在用户程序运行过程中出现系统调用或者发生中断事件,就要运行操作系统(即核心)程序,进程的运行模式就变为 ...

  8. exec系列函数详解

    execve替换进程映像(加载程序):execve系统调用,意味着代码段.数据段.堆栈段和PCB全部被替换.在UNIX中采用一种独特的方法,它将进程创建与加载一个新进程映像分离.这样的好处是有更多的余 ...

  9. PHP 使用 curl_* 系列函数和 curl_multi_* 系列函数进行多接口调用时的性能对比

    在页面中调用的服务较多时,使用并行方式,即使用 curl_multi_* 系列函数耗时要小于 curl_* 系列函数. 测试环境 操作系统:Windows x64 Server:Apache PHP: ...

随机推荐

  1. Elasticsearch(ES)集群的搭建

    1. 概述 Elasticsearch(ES)集群支持分片和副本,能够很容易的实现负载均衡.扩容.容灾.高可用. 今天我们就来聊一下,Elasticsearch(ES)集群是如何搭建的. 2. 场景介 ...

  2. Writing in the Science 01

    INTRODUCTION What makes good writing? Good writing communicates an idea clearly and effectively. Goo ...

  3. appium日志

    2020-10-02 00:44:10:672 [Appium] Welcome to Appium v1.16.0 2020-10-02 00:44:10:673 [Appium] Non-defa ...

  4. 『Python』面向对象(一)

    类和对象 类(class)是用来描述具有相同属性(attribute)和方法(method)的对象的集合,对象(object)是类(class)的具体实例.比如学生都有名字和分数,他们有着共同的属性. ...

  5. Mysql 5.7版本,所有的坑,这里都有

    MYSQL5.7版本流程的坑,我这里都有 必须按照如下操作.不按照下面操作,出错误不要怪我哦_ 我们首先在官网下载mysql5.7版本 解压之后,在bin相同目录下创建一个my.ini配置文件里面内容 ...

  6. 从一个舒服的姿势插入 HttpClient 拦截器技能点

    马甲哥继续写一点大前端,阅读耗时5 minute,行文耗时5 Days 今天我们来了解一下如何拦截axios请求/响应? 这次我们举一反三,用一个最舒适的姿势插入这个技能点. axios是一个基于 p ...

  7. 【.Net vs Java? 】 先来看一下Java和C#的数据类型区别。

    新工作.Net和Java都要做,早期也做过一段Java的项目,但没有系统的深入学习过.一直觉得这两门语言估计是最相近的两门语言了,好多代码可以说直接拷过来都不带报错的,但仔细推敲还是有很多的不同. 1 ...

  8. 题解 AVL 树

    link Description 给出一个 \(n\) 个点的 AVL 树,求保留 \(k\) 个点使得字典序最小. \(n\le 5\times 10^5\) Solution 因为我很 sb ,所 ...

  9. 【c++ Prime 学习笔记】第17章 标准库特殊设施

    17.1 tuple类型 tuple是类似pair的模板: pair和tuple的成员类型都可以不相同 pair恰好有两个成员,tuple可有任意数量的成员 按照不同参数数量和类型实例化出的tuple ...

  10. Wireshark 过滤器的使用

    符号 例子 = = tcp.port = = 80 过滤出来TCP包含80端口的数据包 != ip.src != 127.0.0.1 ip的原地址不是127.0.0.1过滤出来 > lp.len ...