fork()和vfork()的区别,signal函数用法,exec()系列函数的用法小结
一:fork()和vfork()的区别:
fork()函数可以创建子进程,有两个返回值,即调用一次返回两个值,一个是父进程调用fork()后的返回值,该返回值是刚刚创建的子进程的ID;另一个是子进程调用fork()后的返回值,该返回值为0。
vfork与fork不同的地方在于:
使用fork()创建子进程时:子进程只是完全复制父进程的资源,并且哪个进程先运行取决于系统的调度算法。
点击(此处)折叠或打开
- int globVar = 5;
- int main(int argc,char *argv[])
- {
- int var = 1,i;
- pid_t pid;
- printf("fork is different with vfork\n");
- pid = fork();
- // pid = vfork();
- printf("%d---------\n",pid);
- switch(pid)
- {
- case 0:
- i = 2;
- while(i-- > 0)
- {
- printf("--------Child process is running\n");
- globVar++;
- var++;
- printf("--------c sleep\n");
- sleep(1);
- }
- printf("--------Clild's globVar = %d,var = %d\n",globVar,var);
- break;
- case -1:
- perror("Process creat faild\n");
- exit(0);
- default:
- i = 3;
- while(i-- > 0)
- {
- printf("Parent process is running\n");
- globVar++;
- var++;
- printf("p sleep\n");
- sleep(1);
- }
- printf("Parent's globVar = %d,var = %d\n",globVar,var);
- exit(0);
- }
- }
- fork is different with vfork
- 14070---------
- Parent process is running
- p sleep
- 0---------
- --------Child process is running
- --------c sleep
- Parent process is running
- p sleep
- --------Child process is running
- --------c sleep
- Parent process is running
- p sleep
- --------Clild's globVar = 7,var = 3
- Parent's globVar = 8,var = 4
使用vfork()创建子进程时:操作系统并不将父进程的地址空间完全复制到子进程,子进程共享父进程的地址空间,并且保证子进程先运行。
点击(此处)折叠或打开
- 将上面的例子vfork()注释去掉,而将fork();注释掉,可以得到与之不同的结果
- fork is different with vfork
- 0---------
- --------Child process is running
- --------c sleep
- --------Child process is running
- --------c sleep
- --------Clild's globVar = 7,var = 3
- 14160---------
- Parent process is running
- p sleep
- Parent process is running
- p sleep
- Parent process is running
- p sleep
- Parent's globVar = 10,var = 32714
二:signal函数,可以用来处理我们系统的一些信号。
typedef void (*sighandler_t)(int);
sighandler_t signal(int signum, sighandler_t handler));
函数的第一个参数指定该信号的值,第二个参数指定对该信号的处理。可以忽略该信号。参数设为(SIG_ING),也可以采用系统默认的方式,参数设置为(SIG_DFL),下面的例子就是忽略我们的输入信号。
点击(此处)折叠或打开
- #include<stdio.h>
- #include<signal.h>
- int main(int argc ,char *argv[])
- {
- signal(SIGINT,SIG_IGN);
- for(;;);
- return 0;
- }
执行结果就是无限循环,我们ctrl+c也无法停止。
三:exec系列函数(execl,execlp,execle,execv,execvp)函数用法总结:
exec系列函数均可以把当前进程替换成一个新进程,保持原有的pid不变。对于以上函数,我按照自己的理解分为三类,当然这不一定是最好的方法,具体看自己了。
1:execl和execlp:我们给它传入相应命令和参数,它就可以脱离当前进程而独立出来。他们两个的不同在于第一个参数传绝对路径还是相对路径,请看下面例子:
点击(此处)折叠或打开
- #include<stdio.h>
- #include<stdlib.h>
- #include<unistd.h>
- int main(int argc,char *argv[])
- {
- printf("entering main process----\n");
- // execl("/bin/ls","ls","-l","/home",NULL);
- // execlp("ls","ls","-l","/home",NULL);
- printf("exiting main process -----\n");
- return 0;
- }
程序的执行结果就是我/home中的内容了,而第10行的内容并不会被打印出来。有没有懂一点点呢,继续;
2:execv和execvp函数:同样,后面有p就是它可以只传入文件名或者命令名就可以执行,不过它的参数可以自己用argv[]传进去。
点击(此处)折叠或打开
- int ret;
- printf("entering main process----\n");
- char *argv[] = {"ls","-l","/home",NULL};
- // ret = execv("/bin/ls",argv);
- ret = execvp("ls",argv);
- if(ret == -1)
- {
- perror("execvp error");
- }
- printf("exiting main process-----\n");
- return 0;
看吧,主要是execv和execvp只能传入两个参数,所以把其他的东西就放在argv[]中了,执行成功返回0,失败返回-1。
3:execle函数,可以获得环境变量,看下面例子:
(注意:test_execle_child 是一个可以打印当前环境变量的函数)
test_execle_child函数如下:
点击(此处)折叠或打开
- extern char ** environ;
- int main(int argc,char *argv[])
- {
- int i ;
- printf("\n\n现在已经进入main函数中的子程序---- pid=%d\n",getpid());
- sleep(2);
- for(i = 0;environ[i] != NULL;i++)
- {
- printf("%s\n",environ[i]);
- }
- return 0;
- }
点击(此处)折叠或打开
- int ret;
- char * const envp[] = {"aa=11","bb=22","cc=33",NULL};
- printf("entering main process----\n");
- // ret = execl("./test_execle_child","test_execle_child",NULL);
- ret = execle("./test_execle_child","test_execle_child",NULL,envp);
- if(ret == -1)
- {
- perror("execle error");
- }
- printf("exiting main process-----\n");
- return 0;
当函数是execle时:输出我们envp中传进去的环境变量,
当函数时execl时:输出我们当前执行进程的环境变量。
阅读(1) | 评论(0) | 转发(0) |
版权声明:本文为博主原创文章,未经博主允许不得转载。
fork()和vfork()的区别,signal函数用法,exec()系列函数的用法小结的更多相关文章
- fork()和vfork()的区别(转载)
fork和vfork 转载 http://coolshell.cn/articles/12103.html 在知乎上,有个人问了这样的一个问题--为什么vfork的子进程里用return,整个程序会挂 ...
- exec系列函数(execl,execlp,execle,execv,execvp)使用
本节目标: exec替换进程映像 exec关联函数组(execl.execlp.execle.execv.execvp) 一,exec替换进程映像 在进程的创建上Unix采用了一个独特的方法,它将进程 ...
- linux系统编程之进程(五):exec系列函数(execl,execlp,execle,execv,execvp)使用
本节目标: exec替换进程映像 exec关联函数组(execl.execlp.execle.execv.execvp) 一,exec替换进程映像 在进程的创建上Unix采用了一个独特的方法,它将进程 ...
- exec系列函数和system函数
一.exec替换进程映象 在进程的创建上Unix采用了一个独特的方法,它将进程创建与加载一个新进程映象分离.这样的好处是有更多的余地对两种操作进行管理.当我们创建 了一个进程之后,通常将子进程替换成新 ...
- fork和vfork的区别
参见百度百科API说明: fork 头文件: #include<unistd.h> #include<sys/types.h> 函数原型: pid_t fork( void); ...
- Linux系统编程(9)—— 进程之进程控制函数exec系列函数
在Linux中,并不存在exec()函数,exec指的是一组函数,一共有6个,分别是: #include <unistd.h> extern char **environ; int exe ...
- 进程基本-进程创建,僵尸进程,exec系列函数
Linux系统中,进程的执行模式划分为用户模式和内核模式,当进程运行于用户空间时属于用户模式,如果在用户程序运行过程中出现系统调用或者发生中断事件,就要运行操作系统(即核心)程序,进程的运行模式就变为 ...
- exec系列函数详解
execve替换进程映像(加载程序):execve系统调用,意味着代码段.数据段.堆栈段和PCB全部被替换.在UNIX中采用一种独特的方法,它将进程创建与加载一个新进程映像分离.这样的好处是有更多的余 ...
- PHP 使用 curl_* 系列函数和 curl_multi_* 系列函数进行多接口调用时的性能对比
在页面中调用的服务较多时,使用并行方式,即使用 curl_multi_* 系列函数耗时要小于 curl_* 系列函数. 测试环境 操作系统:Windows x64 Server:Apache PHP: ...
随机推荐
- ECSHOP任意页面显示指定分类、数量、排序的任意类型文章,包括只显示置顶or普通的文章
1.在需要使用此功能的PHP页面里最后的?>前面添加以下代码,现在以article.php为例子 /** jinmozhe 专业ECSHOP二次开发 * 获得指定分类ID.文章类型.指定数量.排 ...
- 链式调用+对象属性与遍历+this指向+caller/callee
之前的作业: 提示: 在开发的时候尽量在函数内部将作用都给调用好,在外部就能够直接使用 链式调用: 正常这样是不行的,因为没有具体返回值: return 具体的对象,这样的才是链式操作,jquery ...
- 修改CentOS ll命令显示时间格式
临时更改显示样式,当会话结束后恢复原来的样式: export TIME_STYLE='+%Y-%m-%d %H:%M:%S' 永久改变显示样式,更改后的效果会保存下来 修改/etc/profile文件 ...
- 鸿蒙内核源码分析(用栈方式篇) | 程序运行场地谁提供的 | 百篇博客分析OpenHarmony源码 | v20.04
百篇博客系列篇.本篇为: v20.xx 鸿蒙内核源码分析(用栈方式篇) | 程序运行场地谁提供的 | 51.c.h .o 精读内核源码就绕不过汇编语言,鸿蒙内核有6个汇编文件,读不懂它们就真的很难理解 ...
- 4.自定义类加载器实现及在tomcat中的应用
了解了类加载器的双亲委派机制, 也知道了双亲委派机制的原理,接下来就是检验我们学习是否扎实了,来自定义一个类加载器 一. 回顾类加载器的原理 还是这张图,类加载器的入口是c++调用java代码创建了J ...
- 数据库的高可用MHA实验步骤
一.多机互信的步骤 双机互信的步骤 第一步:在master管理服务器上ssh-keygen 在master同一台管理服务器上重新开一个窗口ssh-copy-id 192.168.0.13自己给自己互信 ...
- logstash输出到rabbitmq
场景 将应用日志文件发送到rabbitmq. filebeat 不支持rabbitmq作为输出.因此,需要先将文件由filebeat发送到logstash ,再由logstash 输出到rabbitm ...
- ArrayList和LinkedList、及Vector对比分析
ArrayList和LinkedList 底层结构 两者的差别主要来自于底层的数据结构不同,ArrayList是基于数组实现的,LinkedList是基于双链表实现的. 接口实现 LinkedList ...
- MyBatis概念和”安装“
MyBatis概念 MyBatis的前身就是iBatis,本是apache的一个开源项目,2010年这个项目由apahce sofeware foundation 迁移到了google code,并且 ...
- FastAPI 学习之路(八)路径参数和数值的校验
系列文章: FastAPI 学习之路(一)fastapi--高性能web开发框架 FastAPI 学习之路(二) FastAPI 学习之路(三) FastAPI 学习之路(四) FastAPI 学习之 ...