- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章深入解读Linux进程函数fork(),vfork(),execX()由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
本文研究的主要是Linux进程函数fork(),vfork(),execX()的相关内容,具体介绍如下.
fork函数:创建一个新进程 。
1、fork()成功后,将为子进程申请PCB和用户内存空间。
2、子进程会复制父进程用户空间的所有数据(代码段、数据段、BSS、堆、栈),文件描述符。
3、复制父亲进程PCB中绝大多数信息。
4、虽然子进程复制了文件描述符,而对于文件描述符相关的文件表项(struct file结构),则采用共享的方式。
一个实例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
#include <unistd.h> //fork fuction
#include <fcntl.h> //file operator
#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h> //exit fuction
#include <string.h>
int
main() {
pid_t pid;
int
i=1;
int
status;
char
*ch1=
"hello"
,*ch2=
"world"
,*ch3=
"IN"
;
int
fd;
if
((fd=open(
"fork.txt"
,O_RDWR|O_CREAT,0644))==-1) {
perror
(
"not open"
);
exit
(EXIT_FAILURE);
}
if
(write(fd,ch1,
strlen
(ch1))==-1) {
//write in fork.txt
perror
(
"not write"
);
exit
(EXIT_FAILURE);
}
if
((pid=fork())==-1) {
perror
(
"fork error"
);
exit
(EXIT_FAILURE);
}
else
if
(pid==0) {
//son process
int
i=2;
//change i
printf
(
"child:i=%d\n"
,i);
if
(write(fd,ch2,
strlen
(ch2))==-1)
perror
(
"child write"
);
return
0;
}
else
{
sleep(1);
printf
(
"parent:i=%d\n"
,i);
if
(write(fd,ch3,
strlen
(ch3))==-1)
perror
(
"child write"
);
wait(&status);
return
0;
}
}
|
运行:
1
2
3
4
|
[root@localhost linux]# gcc -o fork fork.c
[root@localhost linux]# ./fork
child:i=2
parent:i=1
|
可以看到在子进程中改变了i的值,然而父进程i仍为1,所以说子进程和父进程有自己的用户空间。而打开所创建的fork.txt可以得到hellowordIN,父子进程共同对一个文件操作写入的数据是不交叉覆盖的,说明父子进程共享文件偏移,一次共享文件表项.
与fork()函数不同,vfork()函数在创建进程是并不复制父进程的地址空间,而是在必要的时候才申请新的存储空间,因此使得vfork()更有效率.
特别注意的是vfork()是共享父进程的代码以数据段.
一个例子
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
|
#include <unistd.h> //fork fuction
#include <fcntl.h> //file operator
#include <sys/types.h>
#include <stdio.h>
#include <stdlib.h> //exit fuction
#include <string.h>
int
i=10;
int
main() {
pid_t pid;
if
((pid=fork())==-1) {
perror
(
"fork error"
);
exit
(EXIT_FAILURE);
}
else
if
(pid==0) {
//son process
i++;
printf
(
"child:i=%d\n"
,i);
_exit(0);
}
else
{
sleep(1);
printf
(
"parent:i=%d\n"
,i);
return
0;
}
}
|
注意:上面的代码中回收子进程用的是_exit(0),如果用return 0;的话它会回收用户空间,因此在父进程调用的时候会出现段错误.
下面是调用输出结果:
1
2
3
4
5
6
7
|
如果以fork()创建则会输出:
[root@localhost linux]# ./fork
child:i=11
parent:i=10
如果改为vfork(),则:
child:i=11
parent:i=11
|
用fork()函数创建紫禁城后,如果希望在当前子进程中运行新的程序,则可以调用execX系列函数。 注意:当进程调用exec函数后,该进程的用户空间资源完全有新程序代替。 这些函数的区别在于:
1、指示新程序的位置是路径还是文件名 2、在使用参数时是使用参数列表哈市使用argv[]数组 3、后缀有l(list)表示使用参数列表,v表示使用argv[]数组 。
具体如下所示:
1
2
3
4
5
6
7
8
9
10
|
#include<unistd.h>
int
execl(
const
char
*pathname,
const
char
*arg0,...
/*(char *) 0 */
);
int
execv(
const
char
*pathname,
char
*
const
argv[]);
int
execle(
const
char
*pathname,
const
char
*arg0,...
/*(char *) 0
,char *const envp[] */
);
int
execve(
const
char
*pathname,
char
*
const
argv[],
char
*
const
envp[]);
int
execlp(
const
char
*filename,
const
char
*arg0,...
/*(char *) 0*/
);
int
execvp(
const
char
*filename,
char
*
const
argv[]);
int
fexecve(
int
fd,
char
*
const
argv[],
char
*
const
evnp[]);
|
一个实例:
1
2
3
4
5
6
7
8
9
10
11
12
|
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int
main(
int
argc ,
char
* argv[]) {
pid_t pid;
if
((pid=fork())==-1)
printf
(
"error"
);
else
if
(pid==0)
execl(
"/bin/ls"
,
"ls"
,
"-l"
,argv[1],(
char
*)0);
else
printf
(
"father ok\n"
);
}
|
运行可以看到在子进程中执行了ls命令.
1
2
|
[yqtao@localhost linux]$ gcc -o exec execX.c
[yqtao@localhost linux]$ ./exec /home father ok
|
//execlp()函数使用 。
1
2
3
4
5
6
|
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int
main(
int
argc ,
char
* argv[]) {
execlp(
"ls"
,
"ls"
,
"-l"
,
"/home"
,(
char
*)0);
}
|
//execv()函数的使用 。
1
2
3
4
5
6
7
|
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int
main(
int
argc ,
char
* argv[]) {
char
* argv1[]={
"ls"
,
"-l"
,
"/home"
,0};
execv(
"/bin/ls"
,argv1);
}
|
ecvp()会从环境变量PATH所指定的目录中查找文件名作为第一个参数,第二个及以后的参数由参数列表,注意最后一个成员必须为NULL 。
1
2
3
4
5
6
7
|
#include <unistd.h>
#include <stdio.h>
#include <sys/types.h>
int
main(
int
argc ,
char
* argv[]) {
char
* argv1[]={
"ls"
,
"-l"
,
"/home"
,0};
execvp(
"ls"
,argv1);
}
|
以上就是本文关于深入解读Linux进程函数fork(),vfork(),execX()的全部内容,希望对大家有所帮助。感兴趣的朋友可以继续参阅本站其他相关专题,如有不足之处,欢迎留言指出。感谢朋友们对本站的支持! 。
原文链接:http://blog.csdn.net/taoyanqi8932/article/details/52778015 。
最后此篇关于深入解读Linux进程函数fork(),vfork(),execX()的文章就讲到这里了,如果你想了解更多关于深入解读Linux进程函数fork(),vfork(),execX()的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
最近做一个项目,由于是在别人框架里开发app,导致了很多限制,其中一个就是不能直接引用webservice 。 我们都知道,调用webserivice 最简单的方法就是在 "引用"
这是SDL2代码的一部分 SDL主函数 int main(int argc,char *argv[]) { ... ... bool quit=false; S
c 中的函数: PHPAPI char *php_pcre_replace(char *regex, int regex_len, ch
我有以下映射: public class SecurityMap : ClassMap { public SecurityMap() {
我在vue-lic3中使用了SCSS,但是有一个奇怪的错误,使用/ deep /会报告错误,我不想看到它。 代码运行环境 vue-cli3 + vant + scss 的CSS /deep/ .van
我在深入阅读 C# 时遇到了这个我能理解的内容: 当它被限制为引用类型时,执行的比较类型完全取决于类型参数被限制为什么。 但是不能理解这个: 如果进一步限制派生自重载 == 和 != 运算符的特定类型
Closed. This question is opinion-based。它当前不接受答案。 想改善这个问题吗?更新问题,以便editing this post用事实和引用来回答。 3年前关闭。
有人可以详细介绍关于自赋值的运算符重载中的 *this 和 const 例如: Class& Class::operator=(const Class& other) { a = other.
在向树中插入新节点时,如何填充闭包表的深度/长度列? ancestor 和 descendant 中的值是来自另一个表的 ID,表示要以树结构排列的页面。 关闭表: ancestor desce
现在我正在阅读“深入了解 C#”。缺少的一件事是完成一章后我可以解决的一系列问题。那会帮助我理解我刚刚学到的概念。 哪里可以找到适合 C#3.0 的问题集? 谢谢 最佳答案 你可以试试LINQ 101
TypeScript 给 JavaScript 扩展了类型的语法,我们可以给变量加上类型,在编译期间会做类型检查,配合编辑器还能做更准确的智能提示。此外,TypeScript 还支持了高级类型用
是否有一个单行代码来获取生成器并生成该生成器中的所有元素?例如: def Yearly(year): yield YEARLY_HEADER for month in range(1, 13)
所以我阅读了一些与“什么是方法组”相关的 StackOverflow 问题以及其他互联网文章,它们在底线都说了同样的话——方法组是“一组重载方法” ". 但是,在阅读 Jon Skeet 的“C# 深
有什么方法可以从子组件中获取子组件吗? 想象一下以下组件树: 应用程序 问题 问题选项(包含复选框) 问题选项(包含复选框) 问题选项(包含复选框) 我想从 App 访问问题选项以选中所有复选框。 参
class_eval 和 instance_eval 在定义方法等情况下是完全可以预测的。我也理解类的实例和类的单例(又名特征类)之间的区别。 但是 我无法弄清楚以下唯一的事情:比方说,出于某些策略目
我想出了如何将符号 rwx 部分读取/转换为 421 个八进制部分,这非常简单。但是当涉及到特殊字符时,我感到非常困惑。我们知道 -r-xr---wx 转换为 0543,但 -r-sr---wt 或
我怀疑我系统的 Java 版本有问题。某些应用程序出现段错误或内存不足或存在链接错误。如果我从源代码安装了 JDK,我会做类似“make test”的事情,看看哪些测试失败了。但是,看起来从源代码构建
如何克隆一个 repo(使用 libgit2 ) 我想做什么git clone确实,但有 libgit2 .我可能要问的是什么 git clone确实很深入。 这是我目前正在做的: 初始化一个repo
00、头痛的JS闭包、词法作用域? 被JavaScript的闭包、上下文、嵌套函数、this搞得很头痛,这语言设计的,感觉比较混乱,先勉强理解总结一下😂😂😂.
我开始玩 lubridate R 中的包。我注意到 now(tzone="EST")计算为: [1] "2015-08-25 13:01:08 EST" 而 now(tzone="PST")导致警告:
我是一名优秀的程序员,十分优秀!