在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. BZOJ3507 [Cqoi2014]通配符匹配 【哈希 + 贪心】

    题目 几乎所有操作系统的命令行界面(CLI)中都支持文件名的通配符匹配以方便用户.最常见的通配符有两个,一个 是星号(""'),可以匹配0个及以上的任意字符:另一个是问号(&quo ...

  2. 刷题总结——doing homework again(hdu1789)

    题目: Problem Description Ignatius has just come back school from the 30th ACM/ICPC. Now he has a lot ...

  3. AC自动机详解 (P3808 模板)

    AC自动机笔记 0.0 前言 哇,好久之前就看了 KMP 和 Trie 树,但是似乎一直没看懂 AC自动机?? 今天灵光一闪,加上之前看到一些博客和视频,瞬间秒懂啊... 其实这个玩意还是蛮好理解的. ...

  4. linux c 正则表达式

    #include <stdio.h> #include <regex.h> #include <mhash.h> int main() { regex_t rgx; ...

  5. 【kindeditor】KindEditor获取多个textarea文本框的值并判断非空

    kindeditor官网:http://kindeditor.net/demo.php 如何获取多个KindEditor中textarea文本框的值,方式很多种(带有HTML标签). var intr ...

  6. 《Programming in C》读书笔记

    该书由美国Seephen G.Kochan著 贾洪峰译,电子工业出版社,来源是九江学院图书馆采购,现在藏于九江学院图书馆逸夫楼. 本书的主要内容: 第一章.基础知识 第二章.编译和运行第一个程序 第三 ...

  7. Keep-Alive 长连接(转载)

    短连接与长连接 通俗来讲,浏览器和服务器每进行一次通信,就建立一次连接,任务结束就中断连接,即短连接.相反地,假如通信结束(如完成了某个HTML文件的信息获取)后保持连接则为长连接.在HTTP/1.0 ...

  8. Day 21 三元表达式、生成器函数、列表解析

    知识点程序: #! /usr/bin/env python # -*- coding: utf-8 -*- # __author__ = "DaChao" # Date: 2017 ...

  9. 更新YUM源后的arning: rpmts_HdrFromFdno: Header V3 RSA/SHA1 Signature, key ID c105b9de: NOKEY错误

    yum源更新后需要导入key值,否则报错如下,无法安装相关的包. Totalsize:42M DownloadingPackages: warning:rpmts_HdrFromFdno:Header ...

  10. 关于[WinError 10054] 远程主机强迫关闭了一个现有的连接。

    之前一直用python实现qq邮箱自动发送,都弄的好好的,然后今天一打开,就出现如题的错误,百度了许多,说,可能发送邮件次数过多,被当作是攻击,建议换个邮箱,换了也不行, 最后用手机给电脑分享Wifi ...