随笔 - 60  文章 - 24 评论 - 1560 trackbacks - 29

.net asp web c# vb VS2005 VS2008 VS2003

    姓名 景春雷
    网名 1-2-3
    生日 1980.2.29
    城市 沈阳
[生活]昨晚看了变形金刚2。场面火爆,观众也火爆,居然买不到7:30的票,只好等9:30那场。PS:高中以后就没去过电影院了。

与我联系

搜索

 

常用链接

我参与的团队

我的标签

随笔分类(59)

随笔档案(57)

文章分类(23)

相册

收藏夹(2)

积分与排名

  • 积分 - 207375
  • 排名 - 223

最新评论

阅读排行榜

评论排行榜

    猴子抬头道:“我有一个梦,我想我飞起时,那天也让开路,我入海时,水也分成两边,众仙诸神,见我也称兄弟,无忧无虑,天下再无可拘我之物,再无可管我之人,再无我到不了之处,再无我做不成之事,再无我战不胜之物。”
                                                                                    ——今何在
                                                                                          摘自《悟空传》
摘要

男程序员勿进。(因为可能女程序员拍砖的力道会小些,俺比较能扛得住……)

并发冲突——当一条虫子遇上两只小鸡会发生什么事情?

当一条虫子遇上两只小鸡会发生什么事情?
可以肯定的是,那条虫子必定会去见上帝啦。
无法确定的是,到底是那虫子的上半截先去见上帝,还是下半截先去见上帝?
你一准儿在想:“我昨天晚上加班到12点,到现在还晕乎乎的。本想到博客园逛逛,可以暂时忘掉那些复杂多变的需求、防不胜防的Bug以及让人迷惑的办公室政治,没想到却遇到了个精神病,在这儿琢磨这种无聊问题。”
相信我,这绝对是性命攸关的重要问题!想想看,如果是虫子的下半截先去见上帝,上帝一准儿会问它:“你叫啥?是怎么死的啊?”,虫子的下半截答道:“我叫虫子甲,是被小鸡乙吃掉的。”于是上帝在他的本子上写到:“虫子甲是被小鸡乙吃掉的。”然后过了一会儿,虫子的上半截也见到了上帝。上帝也问它:“你叫啥?是怎么死的啊?”虫子的上半截答道:“我叫虫子甲,是被小鸡甲吃掉的。”上帝打开他的本子,发现里面已经写了“虫子甲是被小鸡乙吃掉的。”,然而此时虫子的下半截已经不知爬到哪里逍遥去了,无法详细盘问,于是上帝只好选择相信虫子的上半截,把本子上的记录改成“虫子甲是被小鸡甲吃掉的。”
但是如果碰巧是虫子的上半截先去见的上帝,最后在上帝的本子上就会写着“虫子甲是被小鸡乙吃掉的”。
也就是说,这条可怜的虫虫的死最后会算在哪只小鸡的头上,完全是不可预测的!这个问题不但让上帝颇为头痛,也有可能让程序员丢了性命,欲知后事如何,一段广告过后,马上回来!

广告音:“博客园,不是白菜园、不是幼儿园、不是游乐园,更加不是罗卜家园……博客园,程序员的网上家园。”

1-2-3的超级程序

我最近给客户开发了一个非常厉害的程序。
class Program
{
    
static int n = 0;
    
static void foo1()
    {
        
for (int i = 0; i < 1000000000; i++// 10 亿
        {
                
int a = n;
                n 
= a + 1;
        }
        Console.WriteLine(
"foo1() complete n = {0}", n);
    }
    
static void foo2()
    {
        
for (int j = 0; j < 1000000000; j++// 10 亿
        {
                
int a = n;
                n 
= a + 1;
        }
        Console.WriteLine(
"foo2() complete n = {0}", n);
    }
    
static void Main(string[] args)
    {
        foo1();
        foo2();
    }
}

怎么样?只用了40秒钟,这个程序就计算出了把一个初始为0的变量n累加20亿次1,变量n将等于20亿。什么?你说我是白费CPU不干正经事?这有什么,客户喜欢!顺便说一句,我的电脑是8年前买的,用的是赛扬800的CPU。

能更快些么?

可是,客户居然嫌它太慢了,并且威胁说如果不能把它压缩到10秒以内就让我去见上帝。
我的客户怎么这么狠?唉,打从一开始我就觉得这个总喜欢“击地”的山羊胡老头有些眼熟,这下后悔也晚了,用多线程试试吧。

使用多线程

现在知道我的程序为啥要使用两个函数“foo1()”和“foo2()”来实现了吧?因为我早就算到了这个情况,为使用多线程做了准备,现在我只要把“foo1()”和“foo2()”分别用两个线程来执行就可以了。(要不怎么说再好的架构师也比不上一个能掐会算的算命先生呢?)
class Program
{
    
static int n = 0;
    
static void foo1()
    {
        
for (int i = 0; i < 1000000000; i++// 10 亿
        {
                
int a = n;
                n 
= a + 1;
        }
        Console.WriteLine(
"foo1() complete n = {0}", n);
    }
    
static void foo2()
    {
        
for (int j = 0; j < 1000000000; j++// 10 亿
        {
                
int a = n;
                n 
= a + 1;
        }
        Console.WriteLine(
"foo2() complete n = {0}", n);
    }
    
static void Main(string[] args)
    {
        
new Thread(foo1).Start();
        
new Thread(foo2).Start();
    }
}

你说奇怪不奇怪?一下子结果全都不对了,而且每次执行的结果都不一样!在责怪CPU有Bug、内存有毛病、操作系统中了病毒之前,不妨先来分析一下这段代码是如何执行的。
附言:用了多线程之后,程序执行时间是34秒,所以就算结果正确我的小命也一样不保。

把n加1,统共分3步

在上面那个经过特别设计的例子中,把n加1,统共分3步:
第一步,把n的值保存到a中。
第二步,计算a+1的值。
第三步,把a+1的结果保存到n中。
如果你不能确定上面所说的这三步是不是事实,可以看程序的汇编代码(方法是先在VS2005里单步执行,然后使用菜单“调试 > 窗口 > 反汇编 Ctrl+Alt+D”打开反汇编窗口)。下图截取了汇编代码并使用了相应的伪码,涂了不同的底色以备后用。


正如我们所知道的,CPU只有一个,所以所谓的多个线程“并发执行”只不过是把这些线程排好队,然后让他们一个挨着一个地轮流使用CPU,每个线程只使用很短很短的时间。这样对于人类这样反应迟钝的动物来说,就感觉好像有多个线程在“同时”执行。让我们来看看,当第一个线程执行完“把n的值保存到a中”时,时间到!该轮到别的线程执行了,这时会发生什么事情?这时,你会听到Windows大喝一声:“帮我照顾好我七舅姥爷和他三外甥女——”,然后咔嚓一下就把第一个线程暂停了。所以,如果我们的第一、第二个线程的前三次循环以下图所示的顺序来执行是一点也不奇怪的。(黄色底色的代码属于第一个线程,绿色底色的代码属于第二个线程)


现在,第一、第二个线程里面的循环各自执行了3次,n的值是3,而不是我们期望的6。所以,即使我们的电脑里只有一个CPU(还不是双核的),一样会遇到并发冲突的问题。

真有人在Windows里群殴?

听说有线程发生了并发“冲突”,我们都睁大了眼睛,可实际上并没有什么热闹好看——那两个线程并未打得不可开交。它们虽然访问了同一个全局变量n,但是并未给对方或自己造成什么伤害。我们说这两个线程发生了并发冲突,其实想表达的意思不过是“它们做了我们我们不希望看到的重复工作”而已。

好吧,为了活命,我们必须找到防止这两个线程做重复工作——也就是线程同步——的方法。不过在此之前,先来看看在什么情况下不需要操心线程同步的问题。

不需要线程同步的情况

1. 对n的读取、赋值操作用一条汇编语句就能搞定的时候。

我们把程序稍稍改动一下:


如您所见,“n=n+1” 所对应的汇编代码只有一行 “inc dword ptr ds:[01608A60h]”,也就是通过inc指令直接把CPU的cache里的n的值增加1。CPU的catch真是一个方便的发明呀。不过现在高兴还有些早,因为我们还没有考虑多CPU的情况。要知道现在的服务器大多具有2个以上的CPU,就连PC机都是双核(一块芯片里含有2个逻辑CPU并且有两个cache,就跟安装了两块CPU没啥两样)的了。由于CPU在把cache里的n增加1之后,并不会立即把n的值写入到内存中,所以如果我们在安装了2块CPU的计算机上执行上面那段程序,并且假设第一个线程由CPU1来执行,第二个线程由CPU2来执行,那么这两个线程的前3次循环完全有可能像下面这样:


非常不幸地,n的值是3而不是我们期望的6。CPU cache这个方便的发明现在成了烫手的山芋。不过如果你在装有Intel的双核CPU的计算机上运行上面的代码,会发现结果仍然非常正确,似乎上图所示的麻烦事并没有发生,这是为什么呢?这是因为x86架构的CPU非常地道,它在确保cache一致性方面做了很多努力。
====== 2008-5-26更新 ======
今天又在Intel的双核CPU上测试了一下上面那个代码,发现居然不管加不加volatile关键字,结果都不对!!!不知道是不是我对volatile的理解有误,肯请高手指点。用下面的Interlocked.Increment()结果是正确的。没有认真测试我的代码,十分抱歉!
=======================
坏消息是:不是所有的CPU都像x86 CPU这么地道(例如IA64,由于性能等方面的考虑不会在cache一致性方面多做努力);而好消息是:像IA64这样不地道的CPU都提供了volatile read(强制从内存读取)和volatile write(强制写入内存)操作。相应地,.net提供了volatile关键字。所以,我们只要在定义n的时候加上volatile关键字就可高枕无忧了。

附言 在我的赛扬800 CPU上,不管加不加volatile关键字,程序执行的时间都在14.5秒左右。
你可能不喜欢在声明变量的时候使用volatile关键字,因为这样一来不管是不是使用了多线程、不管是读取还是写入n都会被强制刷新内存;而且如果你把n按引用传递给方法,例如写 int.TryParse("123", out n),volatile关键字将失效。所以.net提供了另一种方案:你可以在声明变量的时候不使用volatile关键字,而在读取和写入n的时候使用Thread.VolatileRead(...)和Thread.VolatileWrite(...)这两个静态方法。另外,还有一个Thread.MemoryBarrier() 函数的功能也是将cache中的数据保存到内存中。

你也可以使用更高级别的互锁方法,例如Interlocked.Increment()。当线程调用Interlocked类中的那些互锁方法时,CPU会强制cache的一致性。事实上,所有的线程同步锁(包括Monitor, ReaderWriterLock, Mutex, Semaphore, AutoResetEvent 以及 ManualResetEvent 等)都会在内部调用互锁方法。

这段程序在我的赛扬800 CPU上运行时间为60秒。

2. 你加m我加n,各加各的

消除线程间的共享资源,无疑是个釜底抽薪的办法。


这个方法虽然很酷,但是却不怎么实用。因为我很难防止别的程序员用两个线程执行foo1()。

3. 只有一个线程对n赋值,其它线程只是读取n值,并且不在乎n值是不是最新的

例如下面这段程序,foo1()负责累加n值,foo2()负责读取进度并且将进度显示给用户。用户呢,看进度只不过是想确定foo1()确实在努力工作中,而不是在炒股、聊QQ、上不良网站或写博客而已。


本篇到此结束,Sleep(1千万毫秒)先。下篇将介绍线程同步的方法。

附言 家里养了2只小鸡,白天的时候就把它们放到窗台上晒太阳。有时候,它们发现了好吃的东西,就会你追我赶的,从窗台的一头跑到另一头,然后扑腾两下小翅膀,作刹车状,再抡圆了小爪子飞快地跑回来,好像在开运动会,十分有趣。
看着两团淡黄色的绒球在窗台上啄食,忽感生命是如此美丽,同时又是如此的脆弱和渺小。
祝 身在震区的人都更坚强、好运!

参考文献

Jeffrey Richter, CLR via C#, Second Edition. Microsoft Press, 2006.
Thomas et al, 孙勇等 译, Programming Ruby 中文版。电子工业出版社,2007.


posted on 2008-05-26 08:18 1-2-3 阅读(4167) 评论(72)  编辑 收藏 网摘 所属分类: 白话线程同步系列

FeedBack:
#1楼 2008-05-26 08:33 菌哥      
难道男程序员每个人手里都拿着砖头?
  回复  引用  查看    
#2楼 2008-05-26 08:41 (无名)[未注册用户]
不错
  回复  引用    
#3楼 2008-05-26 08:42 普若伽门      
这个 小鸡啄虫图 不错。
  回复  引用  查看    
#4楼 2008-05-26 08:42 Solog      
楼主的文章有大师风范.~!不论俺知道的还是不知道的.俺都会看.

  回复  引用  查看    
#5楼 2008-05-26 08:43 Solog      
--引用--------------------------------------------------
普若伽门: 这个 小鸡啄虫图 不错。
--------------------------------------------------------
非小鸡也,乃凤雏也

  回复  引用  查看    
#6楼 2008-05-26 08:46 才子£鸣      
有可能如樓上所說!。。。
  回复  引用  查看    
#7楼[楼主] 2008-05-26 08:51 1-2-3      
@菌哥
呵呵,尝试一下让人看了就行骂的摘要。

  回复  引用  查看    
#8楼 2008-05-26 08:51 icmin[未注册用户]
拿着棒槌来了,嘿嘿
  回复  引用    
#9楼[楼主] 2008-05-26 08:52 1-2-3      
@Solog
谢谢夸奖。啥是凤雏啊?

  回复  引用  查看    
#10楼 2008-05-26 09:10 fun[未注册用户]
我以为这里是卖卫生巾的。
  回复  引用    
#11楼 2008-05-26 09:22 lrr_ec      
倡导大家每一个人都要文明发言,10楼是中病毒了吗?
  回复  引用  查看    
#12楼 2008-05-26 09:24 Jeffrey Zhao      
不错不错,不过好像没有看到出自引用的两本书的内容阿。
  回复  引用  查看    
#13楼 2008-05-26 09:25 Wenly      
博客源怎么会有这种人进来
  回复  引用  查看    
男程序员勿进。
冲着这点。我拍,我拍,我拍死你。

  回复  引用    
#15楼 2008-05-26 09:37 农民佳      

  回复  引用  查看    
#16楼 2008-05-26 09:45 layman[未注册用户]
经常看10楼的文章,没想到10楼竟然。。。唉。。。
  回复  引用    
#17楼 2008-05-26 09:47 代震军      
写的真的很不错,我收藏了:)
  回复  引用  查看    
#18楼 2008-05-26 10:01 狼Robot      
楼主请继续.砖头拿着累.
  回复  引用  查看    
#19楼 2008-05-26 10:08 李战      
http://www.cnblogs.com/Emoticons/qface/055243188.gif" alt="" />景春雷,俺记住您的大名了。

大侠收徒弟不?虽然笨了点...http://www.cnblogs.com/Emoticons/qface/055243523.gif" alt="" />

  回复  引用  查看    
#20楼 2008-05-26 10:17 马可香蕉      
不错,LZ说得通俗易懂:)
  回复  引用  查看    
#21楼 2008-05-26 10:25 任力      
看到那摘要心里痒痒的!
非得进来看个究竟!
楼主继续喔~~~~手里的砖头还没扔呢。。。

  回复  引用  查看    
#22楼 2008-05-26 10:26 王孟军!      
说得很细,不错,很好

  回复  引用  查看    
#23楼[楼主] 2008-05-26 10:30 1-2-3      
@Jeffrey Zhao
在“不需要线程同步的情况”那一节里有很多内容都是在《CLR Via C# 第二版》里学的(最后一章)。《Programming Ruby》里面有一个很不错的线程同步的例子,使用累加一个变量来演示并发冲突是跟这本书学的。

  回复  引用  查看    
#24楼 2008-05-26 10:31 朝晖的.net      
偶没明白~~楼主到底要讲什么呢?
是要证明操作系统存在并发冲突还是将如何实现线程同步?
比起证明并发冲突,我感觉将如何避免并发冲突和实现线程同步,对程序开发更有意义。
可能我过于心急了,下一步楼主就会讲了,是吧。
好文要速度,楼主你就辛苦点吧`~

  回复  引用  查看    
#25楼[楼主] 2008-05-26 10:36 1-2-3      
@李战
太客气了。你可是大虾级的人物,这么随便乱客气,很容易让俺高兴得好几天睡不着,岂不罪过?……

  回复  引用  查看    
#26楼 2008-05-26 10:39 A1[未注册用户]
但凡可能发生冲突的并发皆需同步,但凡需同步的并发总执行效率皆低于顺序执行,但各个任务能较早得到执行,所以并发之优势在“发”,而不在于执行。
若是并行,则只在“发”“发”之间产生冲突需要同步时才会有比单发额外损耗。

  回复  引用    
#27楼 2008-05-26 10:43 A1[未注册用户]
汗!漏了几个字,“但凡需同步的并发总执行效率皆低于顺序执行”是针对单CPU而言。
  回复  引用    
#28楼[楼主] 2008-05-26 10:48 1-2-3      
@A1
http://www.cnblogs.com/Emoticons/qface/055243188.gif" alt="" />

  回复  引用  查看    
#29楼 2008-05-26 10:55 Da Vinci      
来的似乎都是男的....LZ的摘要估计就是这个目的hoho
  回复  引用  查看    
#30楼 2008-05-26 10:58 Jeffrey Zhao      
Sleep(1千万毫秒)
10000000ms = 10000s = 2.7小时
嗯嗯,要说话算话,hoho。

  回复  引用  查看    
#31楼 2008-05-26 11:10 Yes!加菲猫      
喜欢这种风格~另BS博主,男程序员勿进的明确含义就是男程序不进一定吃亏,
o(∩_∩)o...哈哈~

  回复  引用  查看    
#32楼[楼主] 2008-05-26 11:11 1-2-3      
@Jeffrey Zhao
还好不是27小时

  回复  引用  查看    
#33楼[楼主] 2008-05-26 11:12 1-2-3      
@Da Vinci
@Yes!加菲猫
http://www.cnblogs.com/Emoticons/qface/055243142.gif" alt="" />

  回复  引用  查看    
#34楼[楼主] 2008-05-26 11:19 1-2-3      
@代震军
@马可香蕉
@任力
@王孟军!
谢谢夸奖!

  回复  引用  查看    
#35楼 2008-05-26 11:24 Da Vinci      
希望今后写一点进程间通信, 或者并行操作方面的算法hoho

  回复  引用  查看    
#36楼 2008-05-26 11:26       
即将进入多核处理器时代(貌似已经进入),处理好多核下的线程问题越来越重要了。。学习了!!!
  回复  引用  查看    
#37楼[楼主] 2008-05-26 11:40 1-2-3      
@Da Vinci
其实本文里面的东东都是我现学现卖的,进程间的同步我还没试过呢,呵呵。

  回复  引用  查看    
#38楼[楼主] 2008-05-26 11:44 1-2-3      
郑重声明:前10楼那个骂我的留言被删除了(可不是我删的哦),结果前11楼现在变成了10楼,结果呢,很不幸地,他就变成了前10楼的替罪羊,吼吼。
这个问题该怎么解决呢?

  回复  引用  查看    
#39楼 2008-05-26 12:11 DeathKnight      
1千万毫秒才3个小时不到唉 楼主神人也。。。
  回复  引用  查看    
#40楼 2008-05-26 12:23 LuChaoShuai      
我是从 CLR Via C# 第二版 中认识的 volatile 关键字
  回复  引用  查看    
#41楼 2008-05-26 12:24 坏人      
Interlocked.Increment那段代码没测试过吧,嘿嘿,写得比较有趣,但可以再稍微精干一些。
  回复  引用  查看    
竟然才是1 呵呵
弄一个小标题比较好,比如“白话并发冲突与线程同步(1) :男程序员勿进”o(∩_∩)o...

  回复  引用    
#43楼 2008-05-26 12:47 Steven Chen      
http://www.cnblogs.com/Emoticons/baimantou/134950908.gif" alt="" /> 我去面壁。。。。。。。。

-------------------------------
我承认我是看了这句话才进来的 “男程序员勿进”

  回复  引用  查看    
#44楼[楼主] 2008-05-26 12:59 1-2-3      
--引用--------------------------------------------------
坏人: Interlocked.Increment那段代码没测试过吧,嘿嘿,写得比较有趣,但可以再稍微精干一些。
--------------------------------------------------------
Interlocked.Increment那段代码是测试过的。有什么不对的地方么?请指教,谢谢!

  回复  引用  查看    
#45楼 2008-05-26 12:59 Inrie(洪晓军)      
如果大家都接受"男程序员勿进"这个建议,访问量不知道有多少? ^_^
  回复  引用  查看    
#46楼 2008-05-26 13:04 BlackCat      
好,不错
  回复  引用  查看    
#47楼 2008-05-26 13:55 阿齐      
好文!要是文章的字号大点就更妙了
  回复  引用  查看    
博主我按你的代码试了下,我的cpu是intel 双核,有如下疑问:
static void Main(string[] args)
{
foo1();
foo2();
}

运行这段时,时间差不多是8秒

static void Main(string[] args)
{
new Thread(foo1).Start();
new Thread(foo2).Start();
}
运行这段时,时间是0点几秒,怎么会一开始就得到时间差,但是计算的时间明显比8秒长很久,得出来的结果也不对

谢谢~~

  回复  引用    
#49楼[楼主] 2008-05-26 14:45 1-2-3      
@初学,疑问
static void Main(string[] args)
{
new Thread(foo1).Start();
new Thread(foo2).Start();
}
> 运行这段时,时间是0点几秒,怎么会一开始就得到时间差
其实运行这段程序的是3个线程:执行Main()的是一个线程,它又创建了另外两个线程执行foo1()和foo2()。那个0点几秒只是创建这两个线程的时间。

  回复  引用  查看    
#50楼 2008-05-26 15:16 狼Robot      
楼主是不是sleep过头了,怎么还不见2出来呢?
  回复  引用  查看    
#51楼 2008-05-26 15:16 狼Robot      
再不出来扔砖头了哈.
  回复  引用  查看    
#52楼[楼主] 2008-05-26 15:24 1-2-3      
@狼Robot
2我还没写呢。
最近感觉比较累。心中苦闷,很是迷茫(说白了就是有些颓废啦),加上又是现学现卖,所以这个2啥时候出可就很不好说啦。

  回复  引用  查看    
#53楼 2008-05-26 16:36 airwolf2026      
不要出来的太快,好文章要像怀孕一样啊.不然俺都没有时间看
  回复  引用  查看    
#54楼 2008-05-26 17:13 雲淡風清      
楼主,麻烦你用winfrm说明下,不要用控制台,就是在线程中操作窗体控件的时候,处理多个线程和控件间的同步问题,貌似那个 invok什么的
  回复  引用  查看    
#55楼[楼主] 2008-05-26 17:19 1-2-3      
@雲淡風清
因为用WinForm的话需要考虑的情况会更复杂一些(例如需要考虑控件对线程同步的要求等等),所以用的控制台。

  回复  引用  查看    
#56楼 2008-05-26 17:22 peace      
UP~~~
  回复  引用  查看    
#57楼 2008-05-26 17:54 狼Robot      
--引用--------------------------------------------------
1-2-3: @狼Robot
2我还没写呢。
最近感觉比较累。心中苦闷,很是迷茫(说白了就是有些颓废啦),加上又是现学现卖,所以这个2啥时候出可就很不好说啦。
--------------------------------------------------------

我们要向李战老师学习:用水云般自在的禅心,书写诗情画意的程序人生...

  回复  引用  查看    
#58楼 2008-05-26 18:05 finfou[未注册用户]
有一点不明白,为什么在单核的情况下使用多线程能获得性能增益,程序里也没有资源竞争的情况,既主线程始终不间断运行的。加上线程管理的代价,预期会更慢一些。
  回复  引用    
#59楼 2008-05-26 21:15 Agan@CN      
mark
  回复  引用  查看    
#60楼 2008-05-26 22:06 zbb[未注册用户]
我真想给你学两年。。。我感觉什么都不会 太愚钝了 你qq多少。。。
  回复  引用    
#61楼 2008-05-26 22:08 皇帝的新装      
好文。
  回复  引用  查看    
#62楼[楼主] 2008-05-27 08:23 1-2-3      
@zbb
别这么客气,我技术也普通,共同学习吧。可以邮件联系我:zahuifan@163.com

  回复  引用  查看    
问下 LZ 那些类似于汇编语言的东西 咋看呢???
  回复  引用    
#64楼[楼主] 2008-05-27 10:26 1-2-3      
@Tony.chen
先在VS2005里单步执行,然后使用菜单“调试 > 窗口 > 反汇编 Ctrl+Alt+D”打开反汇编窗口.
至于汇编语句嘛,其实我也不熟,对照着源代码猜测吧。

  回复  引用  查看    
#65楼 2008-05-27 13:16 求知无傲      
仰望
  回复  引用  查看    
#66楼 2008-06-02 13:24 成长的强强      
怕以后不能看楼主的妙文,特在此做个记号,方便日后不断地欣赏。
  回复  引用  查看    
#67楼 2008-06-03 14:26 长沙小能      
真的很不错,李战如果说要当你徒弟,那岂不是没有我等的位置了
  回复  引用  查看    
#68楼[楼主] 2008-06-03 16:12 1-2-3      
@长沙小能
呵呵,你别听李战瞎客气。我猜他是看到园子里也有和他一样喜欢编故事的人感觉非常高兴吧。

  回复  引用  查看    
#69楼 2008-06-26 10:58 John Rambo      
inc dword ptr ds:[01608A60h]
我觉得好像并不是所有指令都是原子性的
所以发生那种事也可能

  回复  引用  查看    
#70楼 2008-08-09 10:36 guocai[未注册用户]
using System;
using System.Collections.Generic;
using System.Text;
using System.Threading;
namespace ConsoleApplication1
{
class Program
{
static volatile int i = 0;
static void foo1()
{
for (int j = 0; j < 1000000000; j++)
{
i++;
}
Console.WriteLine("foo1={0}",i);
}
static void foo2()
{
for (int j = 0; j < 1000000000; j++)
{
i++;
}
Console.WriteLine("fool1={0}",i);
}
static void Main(string[] args)
{
new Thread( foo1).Start();
new Thread( foo2).Start();
}
}
}
为什么加了volatile 结果还是不正确的我的机器是 4核CPU +4G内存

  回复  引用    
#71楼[楼主] 2008-08-09 11:39 1-2-3      
@guocai
我也不知道呀,不是Intel的问题,就是.net的问题。总之理论上应该好用,可是实际上就是不好用。用Interlocked.Increment()吧,这个是好用的。

  回复  引用  查看    
#72楼 2008-09-11 14:52 John Rambo      
@guocai
这样写肯定是不正确的。因为i++不是原子操作
i++ 就是
volatile read i
inc
volatile write i

  回复  引用  查看    



发表评论

昵称: [登录] [注册]

主页:

邮箱:(仅博主可见)

评论内容:

  登录  注册

[使用Ctrl+Enter键快速提交评论]

0 1192006




相关文章:

相关链接: