免费注册 查看新帖 |

Chinaunix

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

诡异的多核问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2010-09-11 05:48 |只看该作者 |倒序浏览
假设一个文件file1内容为:
ab
a
cc


此时分别在单核机器和SMP机器下执行下面这条命令:
cat file1 | grep -v "cc" > file1

结果是,在单核下,file1的内容将是:
ab
a
在SMP情况下,file1的内容将是:
空的。

哪位能具体给讲讲为啥会出现两种不同的结果?会是kernel引起的么?

论坛徽章:
2
CU十二周年纪念徽章
日期:2013-10-24 15:41:34处女座
日期:2013-12-27 22:22:41
2 [报告]
发表于 2010-09-11 18:44 |只看该作者
大概,大概,大概,大概:

单核时,cat 与 grep 串行运行,grep是得的到输入的。

多核时,cat 与 grep 并行运行,如果grep先运行的话,file1的内容就会被清空,cat的输出就为空了。

论坛徽章:
0
3 [报告]
发表于 2010-09-13 00:10 |只看该作者
大概,大概,大概,大概:

单核时,cat 与 grep 串行运行,grep是得的到输入的。

多核时,cat 与 gre ...
tempname2 发表于 2010-09-11 18:44


不能这么说,如果这样解释,那多核和SMP还有意义吗?

问题确实比较诡异,我也没想明白

论坛徽章:
0
4 [报告]
发表于 2010-09-13 11:02 |只看该作者
回复 3# myguyan


    为什么说这样子多核就没有意义了呢?

    两个“同时”运行的进程,一个要读文件、另一个要清空文件,并且两者之间没有任何同步措施。最后到底文件是先被读还是先被清空,这个是没有保证的。

论坛徽章:
7
丑牛
日期:2013-10-18 14:43:21技术图书徽章
日期:2013-11-03 09:58:03辰龙
日期:2014-01-15 22:57:50午马
日期:2014-09-15 07:04:39丑牛
日期:2014-10-16 14:25:222015年亚洲杯之伊朗
日期:2015-03-16 10:24:352015亚冠之城南
日期:2015-05-31 09:52:32
5 [报告]
发表于 2010-09-13 11:59 |只看该作者
回复 4# kouu


    如果在多核下不能保证.那么在单核下可内核抢占时也没有保证了

论坛徽章:
0
6 [报告]
发表于 2010-09-13 12:28 |只看该作者
为什么要保证呢? 两个进程,各干各的事,有要求内核提供什么保证吗? 没有吧…… 那么还保证什么?

论坛徽章:
36
IT运维版块每日发帖之星
日期:2016-04-10 06:20:00IT运维版块每日发帖之星
日期:2016-04-16 06:20:0015-16赛季CBA联赛之广东
日期:2016-04-16 19:59:32IT运维版块每日发帖之星
日期:2016-04-18 06:20:00IT运维版块每日发帖之星
日期:2016-04-19 06:20:00每日论坛发贴之星
日期:2016-04-19 06:20:00IT运维版块每日发帖之星
日期:2016-04-25 06:20:00IT运维版块每日发帖之星
日期:2016-05-06 06:20:00IT运维版块每日发帖之星
日期:2016-05-08 06:20:00IT运维版块每日发帖之星
日期:2016-05-13 06:20:00IT运维版块每日发帖之星
日期:2016-05-28 06:20:00每日论坛发贴之星
日期:2016-05-28 06:20:00
7 [报告]
发表于 2010-09-13 15:59 |只看该作者
cat file1 | grep -v "cc" > file1

把这条命令改成:
cat file1 | grep -v "cc" > file2

再试试。

论坛徽章:
7
丑牛
日期:2013-10-18 14:43:21技术图书徽章
日期:2013-11-03 09:58:03辰龙
日期:2014-01-15 22:57:50午马
日期:2014-09-15 07:04:39丑牛
日期:2014-10-16 14:25:222015年亚洲杯之伊朗
日期:2015-03-16 10:24:352015亚冠之城南
日期:2015-05-31 09:52:32
8 [报告]
发表于 2010-09-13 18:33 |只看该作者
回复 6# kouu


    我是说如果多核不能保证.那么单核也不能保证.
既然这样.那么于核无关.就是在任何情况下,2个程序的先后无保证.
那我们还敢用SHELL做什么呢?
这个代码用了管道.
想下fork grep后程序后必是先把输入输出描述符准备好.然后在while处理一直到输入没有为止或者被关闭为止.处理的就往输出里写.
如果先后无保证,即你说的2个程序并行.可能一开始就不能初始化好输入.因为可能CAT 还没准备好输出描述符号.
所以在fork grep前至少保证了cat进程已经将file打开了. 这个过程应该有shell实现来保证.

论坛徽章:
0
9 [报告]
发表于 2010-09-13 18:56 |只看该作者
本帖最后由 kouu 于 2010-09-13 19:08 编辑
所以在fork grep前至少保证了cat进程已经将file打开了. 这个过程应该有shell实现来保证.
smalloc 发表于 2010-09-13 18:33


我和你的想法不一样。我的理解是,shell只管fork两个进程,利用管道重定向它们的fd,然后分别execve执行"cat file1"和"grep -v "cc" > file1"。
在第一个execve里面,file1只是cat这个应用程序的参数,shell怎么知道它什么时候open(file1)? shell根本就不知道cat程序要把file1参数当作文件来打开。那么shell又怎么保证fork grep前cat进程已经将file打开了?shell要做的事情只是运行cat,并把file1作为参数传到cat的main函数里面。要不要open(file1),那是cat的事情。

至于grep -v "cc" > file1,这里可能有些纠结。我之前的理解是:shell可能会在fork出执行grep的进程之前就把file1打开了,并重定向grep的输出。如果是这样的话,在cat和grep开始之前,file1就已经被清空了。
但是在我和环境里面,shell的执行流程是:(用strace观察的)
1、pipe创建管道
2、clone两个子进程(两个子shell),并使用管道做重定向
3、两个子shell分别去执行cat file1和grep -v "cc" > file1
这样的话,cat file1执行的时候,file1的确有可能还有被执行grep的那个子shell打开的。

论坛徽章:
0
10 [报告]
发表于 2010-09-13 19:39 |只看该作者
仔细想了一下,好像明白 smalloc  兄的意思了。
“所以在fork grep前至少保证了cat进程已经将file打开了“,smalloc兄的理解是不是:
1、shell应该顺序执行管道符相连的程序;
2、当前一个程序有输出以后,才会fork第二个进程,以执行第二个程序;

我的理解不是这样的,我认为shell是一下子就fork两个子进程,然后分别执行两个程序。具体第一个程序什么时候write、第二个程序什么时候read,shell不管。
执行sleep 100 | echo 123试一试,123会立马输出,而不是等到100秒以后。
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

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

清除 Cookies - ChinaUnix - Archiver - WAP - TOP