免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 1465 | 回复: 2
打印 上一主题 下一主题

[Linux] 关于vfork函数[已解决] [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2013-11-11 20:17 |只看该作者 |倒序浏览
本帖最后由 crazyhadoop 于 2013-11-13 22:21 编辑

程序1:

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>

  4. int g_a = 1;

  5. int main(void){
  6.         int l_b = 1;
  7.         pid_t p;
  8.         if((p = vfork()) < 0){
  9.                 perror("fork");
  10.         }else if(p == 0){
  11.                 g_a++;
  12.                 l_b++;
  13.         }
  14.         printf("ppid = %d,pid = %d,g_a = %d,l_b = %d\n", getppid(), getpid(), g_a, l_b);
  15.         return 0;
  16. }
复制代码
代码执行时出现段错误:
yan@yan-vm:~/apue$ ./a.out
ppid = 4101,pid = 4102,g_a = 2,l_b = 2
ppid = 3791,pid = 4101,g_a = 2,l_b = 1
Segmentation fault (core dumped)

如果在子进程中加入exit或者_exit函数后就能成功运行。
程序2:

  1. #include <stdio.h>
  2. #include <unistd.h>
  3. #include <stdlib.h>

  4. int g_a = 1;

  5. int main(void){
  6.         int l_b = 1;
  7.         pid_t p;
  8.         if((p = vfork()) < 0){
  9.                 perror("fork");
  10.         }else if(p == 0){
  11.                 g_a++;
  12.                 l_b++;
  13.                 exit(0);
  14.         }
  15.         printf("ppid = %d,pid = %d,g_a = %d,l_b = %d\n", getppid(), getpid(), g_a, l_b);
  16.         return 0;
  17. }
复制代码
yan@yan-vm:~/apue$ ./a.out
ppid = 3791,pid = 4115,g_a = 2,l_b = 2

在程序1中子进程最后执行了return 0,在程序2中子进程最后执行了exit(0),这两个语句的效果不应该是一样的么?如果不一样,区别是什么?

论坛徽章:
16
CU十二周年纪念徽章
日期:2013-10-24 15:41:3415-16赛季CBA联赛之广东
日期:2015-12-23 21:21:55青铜圣斗士
日期:2015-12-05 10:35:30黄金圣斗士
日期:2015-11-26 20:42:16神斗士
日期:2015-11-19 12:47:50每日论坛发贴之星
日期:2015-11-18 06:20:00程序设计版块每日发帖之星
日期:2015-11-18 06:20:002015亚冠之城南
日期:2015-11-10 19:10:492015亚冠之萨济拖拉机
日期:2015-10-28 18:47:282015亚冠之柏太阳神
日期:2015-08-30 17:21:492015亚冠之山东鲁能
日期:2015-07-07 18:48:39摩羯座
日期:2014-08-29 23:01:42
2 [报告]
发表于 2013-11-11 23:31 |只看该作者
vfork() differs from fork(2) in that the parent is suspended until  the
       child  terminates (either normally, by calling _exit(2), or abnormally,
       after delivery of a fatal signal), or it makes  a  call  to  execve(2).
       Until  that point, the child shares all memory with its parent, includ-
       ing the stack.  The child must not return from the current function  or
       call exit(3), but may call _exit(2)
慢 vfork的一段说明  我也不知道为啥,,你好好研究哈 研究完了贴出答案来,望

论坛徽章:
0
3 [报告]
发表于 2013-11-13 20:40 |只看该作者
谈一点自己的陋见:
你将vfork同fork的用法等同起来了,其实他们的差别挺大的,如tc1989tc兄说。vfork同父进程××共用××内存空间。vfork的代码执行的时候,父进程是停止的,不然父子进程就会使用同样的内存空间,就会相互干扰。只有当子进程执行完毕退出时,他会向父进程发送SIGCHILD信号告知父进程,父进程才会继续运行。但是如果子进程如果不是通过这几种方式结束(by calling _exit(2), or abnormally,  after delivery of a fatal signal), or it makes  a  call  to  execve(2).)就会从同父进程共用的代码中返回,也就是从main中返回,这时就会操作main的栈(因为父子的内存空间是共用的),子进程可以很好的退出。但是当父进程也从main退出的时候,其栈上的内容,特别是返回地址被子进程返回的时候修改了。因此从main返回后从栈中取出的返回地址是一个不可预测的地址,这时一般就会产生内存访问错误。

可以参见:
http://bbs.chinaunix.net/thread-3626687-1-1.html
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP