在coolshell.cn上看到的一个问题。为此拿来研究一下。

首先 看看return和exit的差别

在linux上分别跑一下这个代码

int main()
{
return 0;
//exit(0);
}

return 0



exit(0)





return在返回时会对栈进行操作,将栈清空。然后跳转到调用函数的下一条指令。而exit没有对栈操作。详细exit怎么执行。会在linux0.12内核里面研究。

看看以下一段代码

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
int main(void) {
int var;
int pid;
var = 88;
if ((pid = vfork()) < 0) {
printf("vfork error");
exit(-1);
} else if (pid == 0) { /* 子进程 */
var++;
return 0;
}
printf("pid=%d, var=%d\n", getpid(), var);
return 0;
}

执行之后



非常难想象

假设将return 换成 exit



就执行正常。

奇妙的非常。为此分析一下。

在看看使用的这个函数vfork

假设改成fork会是什么样

例如以下

fork+exit



执行正常

fork+return



也是能够正常执行的,

所以还要研究一下fork和vfork

man fork

Historic description

Under Linux, fork(2) is implemented using copy-on-write pages, so the

only penalty incurred by fork(2) is the time and memory required to

duplicate the parent’s page tables, and to create a unique task

structure for the child. However, in the bad old days a fork(2)

would require making a complete copy of the caller’s data space,

often needlessly, since usually immediately afterward an exec(3) is

done. Thus, for greater efficiency, BSD introduced the vfork()

system call, which did not fully copy the address space of the parent

process, but borrowed the parent’s memory and thread of control until

a call to execve(2) or an exit occurred. The parent process was

suspended while the child was using its resources. The use of

vfork() was tricky: for example, not modifying data in the parent

process depended on knowing which variables were held in a register.

有两点比較重要

1、

the vfork()

system call, which did not fully copy the address space of the parent

process, but borrowed the parent’s memory and thread of control until

a call to execve(2) or an exit occurred.

2、

The parent process was

suspended while the child was using its resources.

也就是vfork是和父进程共享内存的,这里包含栈空间。当调用execve或者exit才会停止这样的共享,而且父进程在子进程执行时是挂起的。

试验一下

測试fork是不是copy_on_write,而且两个进程是同一时候执行的

改动为



执行结果为



能够看出。var两个进程不共享,且两个进程执行并无先后顺序

相同上面那段代码,再改动为 vfork

执行



始终是son先执行,然后exit之后父进程再执行,而且父进程执行时。依然共享var的值

所以到此这个问题也就能够就是了

vfork是子进程共享父进程栈。而return指令是清空栈。返回调用main的函数。而exit不会清理栈的内容。

子进程通过return返回后。父进程会去执行,而此时栈里面已经清空。所以会出问题。

为什么会出项执行多次父进程。网上有人讲,是有的os会又一次调用main。

linux之return和exit引发的大问题(vfork和fork)的更多相关文章

  1. Linux编程return与exit区别

    Linux编程return与exit区别 exit  是用来结束一个程序的执行的,而return只是用来从一个函数中返回. return return 表示从被调函数返回到主调函数继续执行,返回时可附 ...

  2. return和exit函数的区别

    在上Linux课的时候,老师提到一句,调用vfork产生的子进程就是为了使用exec族函数来执行其他的代码逻辑. 在子进程退出的时候有两种方式,exit和exec族函数,不能使用return,为什么不 ...

  3. 关于return和exit

    关于return和exit 在子进程退出的时候有两种方式,exit和exec族函数,不能使用return,为什么不能用return呢,exit改成return 会出现父子进程又各自重复开始进行. 1. ...

  4. 理解 break, continue, return 和 exit

    你们知道 “break”, “continue”, “return” 和 “exit”的作用吗? 它们是功能强大的语言结构体.下面通过一个测试函数来说明它们之间的不同. 1 2 3 4 5 6 7 8 ...

  5. DevC++ return 1 exit status

    当使用DevC++时编译运行程序出现 return 1 exit status 有可能是因为有在运行的命令窗口未关闭.

  6. Windows下return,exit和ExitProcess的区别和分析

    通常,我们为了使自己的程序结束,会在主函数中使用return或调用exit().在windows下还有ExitProcess()和TerminateProcess()等函数. 本文的目的是比较以上几种 ...

  7. C++中的return和exit区别

    在main函数中,return和exit经常混用,两者的一个区别:return会执行statck unwinding,而exit不会.如果触发了信号,exit也同样不会做stack unwinding ...

  8. oracle 存储过程循环体中的return和exit区别:

    oracle 存储过程循环体中的return和exit区别:   (1) return 跳出整个循环,终止该循环, 后面的不再执行.     相当于 Java 中的break;   (2)  exit ...

  9. return 与 exit() 的区别

    return是一个关键字,返回函数值:exit()是一个函数: return是语言级的:exit()是操作系统提供的函数: return表示函数退出:exit()表示进程退出: 非主函数中调用retu ...

随机推荐

  1. PHP curl 封装 GET及POST方法很不错的

    <?php function curl_get($url, array $params = array(), $timeout = 5) { $ch = curl_init(); curl_se ...

  2. Linux Shell系列教程之(八)Shell printf命令详解

    本文是Linux Shell系列教程的第(八)篇,更多shell教程请看:Linux Shell系列教程 在上一篇:Linux Shell系列教程之(七)Shell输出这篇文章中,已经对Shell p ...

  3. HDU——2093考试排名(string类及其函数的运用以及istringstream)

    考试排名 Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Submi ...

  4. MockMvc详解

    ★  MockMvc - SpringMVC单元测试的独立测试: 一.简介 为何使用MockMvc?         对模块进行集成测试时,希望能够通过输入URL对Controller进行测试,如果通 ...

  5. net7:Web用户控件ascx的使用及其动态加载

    原文发布时间为:2008-07-30 -- 来源于本人的百度文章 [由搬家工具导入] Web用户控件test.ascx的源代码: using System;using System;using Sys ...

  6. es6总结(三)--数组

  7. 标准C程序设计七---114

    Linux应用             编程深入            语言编程 标准C程序设计七---经典C11程序设计    以下内容为阅读:    <标准C程序设计>(第7版) 作者 ...

  8. gitlab 安装遇到 fatal:does not appear to be a git repository fatal: Could not read from remote repository. 问题

    Cloning into 'door_lock_bsp'... git@192.168.1.5's password:  fatal: 'door_lock/door_lock_bsp.git' do ...

  9. 洛谷——P2149 [SDOI2009]Elaxia的路线

    P2149 [SDOI2009]Elaxia的路线 题目描述 最近,Elaxia和w的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间.Elaxia和w每 ...

  10. SPFA 求带负权的单源最短路

    int spfa_bfs(int s) { ///s表示起点. queue <int> q; memset(d,0x3f,sizeof(d)); ///d数组中存下的就是最短路径(存在的话 ...