设为首页收藏本站Access中国

Office中国论坛/Access中国论坛

 找回密码
 注册

QQ登录

只需一步,快速开始

返回列表 发新帖
查看: 1583|回复: 0
打印 上一主题 下一主题

C#多线程同步

[复制链接]
跳转到指定楼层
1#
发表于 2007-6-8 06:38:00 | 只看该作者 回帖奖励 |倒序浏览 |阅读模式
在编写多线程程序时无可避免会遇到线程的同步问题。什么是线程的同步呢



举个例子:如果在一个公司里面有一个变量记录某人T的工资count=100,有两个主管A和B(即工作线程)在早一些时候拿了这个变量的值回去



,过了一段时间A主管将T的工资加了5块,并存回count变量,而B主管将T的工资减去3块,并存回count变量。

好了,本来T君可以得到102块的工资的,现在就变成98块了。这就是线程同步要解决的问题。



在.Net的某些对象里面,在读取里面的数据的同时还可以修改数据,这类的对象就是“线程安全”。

但对于自己编写的代码段而言,就不需使用线程同步技术来保证数据的完整性和正确性了。



记住这点:

1、如果一个对象(或变量)不会同时被多个其他线程访问,那么这个对象是不需使用线程同步的。

2、如果虽然有多个线程同时访问一个对象,但他们所访问的数据或方法并不相同(不交叉),那这种情况也不需使用线程同步。

例如上例中的那个公司里面如果有 T 和 Q 两个人,但他们的工资分别是由 A 和 B 主管的,那么这个工资的处理就不需要线程同步了。

3、如果一个对象会同时被多个其他线程访问,一般只需为这个对象添加线程同步的代码,而其他线程是不需添加额外代码的。



在C#里面用于实现线程同步的常用类有如下几类

1、Mutex类(互斥器),Monitor类,lock方法

2、ManualResetEvent类,AutoResetEvent类(这两个都是由EventWaitHandle类派生出来的)

3、ReaderWriterLock类



同一类的作用都差不多:其中

第一类的作用是:用来保护某段代码在执行的时候以独占的方式执行,这时如果有第二个线程想访问这个对象时就会被暂停。一直等到独占的



代码执行为止。使用这个方法就可以解决文章一开始时提出的问题了。例如主管A要处理T君的工资之前,先lock一下T君,然后取出目前的



count值,处理完之后再解除T君的锁定。如果在主管B在主管A处理工资时也想取出count值,那么它只能是一直地等待A处理完之后才能继续了





使用这个方法的一个缺点就是会降低程序的效率。本来是一个多个线程的操作,一旦遇到lock的语句时,那么这些线程只要排队处理,形同一



个单线程操作。



下面举个例子说明一下这三个方法的使用:

假定有一个Tools类,里面一个int变量,还有Add和Delete方法,其中Add方法会使int变量的值增加,Delete方法使int变量值减少:



public class Tools

{



private int count = 100;

public void Add(int n)

{

count =n;

}



public void Delete(int n)

{

count-=n;

}

}



在多个线程同时访问这段代码的情况下,因为一个语句会被编译器编译成多个指令,所以会可能出现这种情况:但某个线程调用Add方法时,这



时的count值为100,而正当要加上n的时候,另外一个线程调用了Delete,它要减去m,结果count加上了n,然后又在原先count=100的值的情况



下减掉了m,最后的结果是count被减去了m,而没有加上n。

很明显Add方法和Delete方法是不能同时被调用的,所以必须进行线程同步处理。简单的方法是用lock语句:



public class Tools

{

private object abcde = new object();

private int count = 100;



public void Add(int n)

{

lock(abcde)

{

count =n;

}

}



public void Delete(int n)

{

lock(abcde)

{

count-=n;

}

}

}



其中abcde是一个private级的内部变量,它不表示任何的意义,只是作为一种“令牌”的角色。

当执行Add方法中的lock(abcde)方法时,这个令牌就在Add方法的手中了,如果这时有第二个线程也想拿这个令牌,没门,惟有等待。一旦第一



个lock语句的花括号范围结束之后,这时令牌就被释放了,同时会迅速落到第二个线程的手中,并且排除其他后来的人。



使用Monitor类的方法大致一样:



public class Tools

{

private object abcde = new object();

private int count = 100;



public void Add(int n)

{

Monitor.Enter(abcde);

count =n;

Monitor.Exit(abcde);

}



public void Delete(int n)

{

Monitor.Enter(abcde);

count-=n;

Monitor.Exit(abcde);

}

}



Monitor的常用方法:Enter和Exit都是静态方法,作用跟lock语句的两个花括号一样。



而使用 Mutex 就不需声明一个“令牌”对象了,但要实例化之后才可以使用:



public class Tools

{

private Mutex mut = new Mutex();

private int count = 100;



public void Add(int n)

{

mut.WaitOne();

count =n;

mut.ReleaseMutex();

}



public void Delete(i
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏 分享分享 分享淘帖 订阅订阅
您需要登录后才可以回帖 登录 | 注册

本版积分规则

QQ|站长邮箱|小黑屋|手机版|Office中国/Access中国 ( 粤ICP备10043721号-1 )  

GMT+8, 2024-6-22 22:42 , Processed in 0.083539 second(s), 24 queries .

Powered by Discuz! X3.3

© 2001-2017 Comsenz Inc.

快速回复 返回顶部 返回列表