C#中的几个线程同步对象方法

在编写多线程程序时无可避免会遇到线程的同步问题。什么是线程的同步呢?

举个例子:如果在一个公司里面有一个变量记录某人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(int n) 

mut.WaitOne(); 
count-=n; 
mut.ReleaseMutex(); 

}

其中的WaitOne为等待方法,一直等到Mutex 被释放为止。初始的情况下,Mutex 对象是处于释放状态的,而一旦执行了WaitOne方法之后,它 
就被捕获了,一直到被调用了ReleaseMutex方法之后才被释放。 
使用这三种方法都有一个要注意的问题,就是在独占代码段里面如果引起了异常,可能会使“令牌”对象不被释放,这样程序就会一直地死等下去了。 
所以要在独占代码段里面处理好异常。例如下面这样的代码就是错误的:

public void Add(int n) 

try 

mut.WaitOne(); 
count+=n; 
//....这里省略了N行代码 
//....这里是有可能引起异常的代码 
//....这里省略了N行代码 
mut.ReleaseMutex(); 

catch 

Console.Writeline("error."); 

}

上面的代码一旦在try和catch里面发生了异常,那么Mutex就不能被释放,后面的程序就会卡死在WaitOne()一行,而应该改成这样:

public void Add(int n) 

mut.WaitOne(); 
try 

count+=n; 
//....这里省略了N行代码 
//....这里是有可能引起异常的代码 
//....这里省略了N行代码 

catch 

Console.Writeline("error."); 

mut.ReleaseMutex(); 
}

现在谈一下第二种: 
ManualResetEvent类,AutoResetEvent类

上面这两个类都是由EventWaitHandle类派生出来的,所以功能和调用方法都很相似。 
这两个类常用于阻断某个线程的执行,然后在符合条件的情况下再恢复其执行。 
举个例子,你想送花给一个MM,托了一个送花的小伙子送了过去,而你希望当MM收到花之后就立即打个电话过去告诉她。

但问题是你不知道花什么时候才送到MM的手里,打早了打迟了都不好,这时你可以使用ManualResetEvent对象帮忙。当委

托小伙子送花过去的时候,使用ManualResetEvent的WaitOne方法进行等待。当小伙子把花送到MM的手中时,再调用一下

ManualResetEvent的Set方法,你就可以准时地打电话过去了。 
另外ManualResetEvent还有一个Reset方法,用来重新阻断调用者执行的,情况就好比你委托了这个小伙子送花给N个MM,

而又想准时地给这N个MM打电话的情况一样。

using System; 
using System.Threading;

public class TestMain 

private static ManualResetEvent ent = new ManualResetEvent(false);

public static void Main() 

Boy sender = new Boy(ent); 
Thread th = new Thread(new ThreadStart(sender.SendFlower)); 
th.Start();

ent.WaitOne(); //等待工作 
Console.WriteLine("收到了吧,花是我送嘀:)"); 
Console.ReadLine(); 
}

}

public class Boy 

ManualResetEvent ent;

public Boy(ManualResetEvent e) 

ent = e; 
}

public void SendFlower() 

Console.WriteLine("正在送花的途中"); 
for (int i = 0; i < 10; i++) 

Thread.Sleep(200); 
Console.Write(".."); 

Console.WriteLine(" 花已经送到MM手中了,boss");

ent.Set(); //通知阻塞程序 

}

而AutoResetEvent类故名思意,就是在每次Set完之后自动Reset。让执行程序重新进入阻塞状态。
即AutoResetEvent.Set() 相当于 ManualResetEvent.Set() 之后又立即 ManualResetEvent.Reset(),
其他的就没有什么不同的了。 
举个送花给N个MM的例子:

using System; 
using System.Threading;

public class TestMain 

private static AutoResetEvent ent = new AutoResetEvent(false);

public static void Main() 

Boy sender = new Boy(ent);

for (int i = 0; i < 3; i++) 

Thread th = new Thread(new ThreadStart(sender.SendFlower)); 
th.Start(); 
ent.WaitOne(); //等待工作 
Console.WriteLine("收到了吧,花是我送嘀:) "); 
}

Console.ReadLine(); 
}

}

public class Boy 

AutoResetEvent ent;

public Boy(AutoResetEvent e) 

ent = e; 
}

public void SendFlower() 

Console.WriteLine("正在送花的途中"); 
for (int i = 0; i < 10; i++) 

Thread.Sleep(200); 
Console.Write(".."); 

Console.WriteLine(" 花已经送到MM手中了,boss");

ent.Set(); //通知阻塞程序,这里的效果相当于 ManualResetEvent的Set()方法+Reset()方法 

}

时间: 2024-10-08 00:18:59

C#中的几个线程同步对象方法的相关文章

.NET中的异步操作及线程同步

执行异步操作 CLR使用了WIN的线程处理能力,但保留了与其分离的权利.某些时候CLR的线程与Win的线程不是完全的匹配. 线程的系统开销较大,应限制其数量. 创建:分配并初始化一线程内核对象,保留1M的地址空间(用户模式),12KB的堆栈(内核模式),然后调用DLL函数通知进程中所有DLL操作来通知所有DLL操作又有一新的线程了. 销毁:进程中所有DLL收到一个死亡通知,且释放资源.且当仅有一个CPU时,每隔20ms执行一次上下文切换. CLR线程池. 每个进程一个,被该进程内的所有APP域共

Java并发编程:Java中的锁和线程同步机制

锁的基础知识 锁的类型 锁从宏观上分类,只分为两种:悲观锁与乐观锁. 乐观锁 乐观锁是一种乐观思想,即认为读多写少,遇到并发写的可能性低,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,采取在写时先读出当前版本号,然后加锁操作(比较跟上一次的版本号,如果一样则更新),如果失败则要重复读-比较-写的操作.Java中的乐观锁基本都是通过CAS操作实现的,CAS是一种更新的原子操作,比较当前值跟传入值是否一样,一样则更新,否则失败. 悲观

同步方法 sleep和wait 线程同步的方法

当一个线程进入一个对象的一个synchronized方法后,其它线程是否可进入此对象的其它方法? 分两种情况 1):进入此对象的非同步方法 答案:可以 2):进入此对象的同步方法 答案:不可以 sleep指线程被调用时,占着CPU不工作,形象地说明为"占着CPU睡觉",此时,系统的CPU部分资源被占用,其他线程无法进入,会增加时间限制.wait指线程处于进入等待状态,形象地说明为"等待使用CPU",此时线程不占用任何资源,不增加时间限制.所以sleep(100L)意

JAVA中线程同步的方法

用什么关键字修饰同步方法 ? 用synchronized关键字修饰同步方法  同步有几种实现方法,都是什么?分别是synchronized,wait与notify wait():使一个线程处于等待状态,并且释放所持有的对象的lock.sleep():使一个正在运行的线程处于睡眠状态,是一个静态方法,调用此方法要捕捉InterruptedException异常.notify():唤醒一个处于等待状态的线程,注意的是在调用此方法的时候,并不能确切的唤醒某一个等待状态的线程,而是由JVM确定唤醒哪个线

JAVA中线程同步的方法(7种)汇总

一.同步方法 即有synchronized关键字修饰的方法. 由于java的每个对象都有一个内置锁,当用此关键字修饰方法时, 内置锁会保护整个方法.在调用该方法前,需要获得内置锁,否则就处于阻塞状态. 注: synchronized关键字也可以修饰静态方法,此时如果调用该静态方法,将会锁住整个类. 二.同步代码块 即有synchronized关键字修饰的语句块. 被该关键字修饰的语句块会自动被加上内置锁,从而实现同步 代码如: synchronized(object){ } 注:同步是一种高开销

线程/同步对象的属性对象

SysConf函数检查系统的能力 1.线程属性对象 线程创建时,可以初始化一个线程属性对象,对应的有线程属性对象的回收函数 线程属性对象的线程分离属性,如果不需要获取线程终止状态 设置线程栈的位置和大小 线程栈的警戒区 2.同步对象的属性对象 ----互斥量同步对象的属性对象: 1)互斥量进程共享属性----互斥量进程间同步 2)互斥量进程共享健壮性---默认是一个持有锁的进程异常终止时,会拖住其它的进程, 通过设置健壮性属性,加锁的函数返回值会有三个而不是两个,可以判断出终止进程没有释放锁的情

Delphi线程同步的方法

更详细的可以参考:http://www.cnblogs.com/xumenger/p/4450659.html 或者参考之后的博客 四个系统内核对象(事件.互斥.信号.计时器)都是线程同步的手段,从这也能看出处理线程同步的复杂性:不过这还不是全部,Windows Vista开始增加了 Condition variables(条件变量).Slim Reader-Writer Locks(读写锁)等同步手段. 不过最简单.最轻便(速度最快)的同步手段还是 CriticalSection(临界区),但

python中的类方法、静态方法、对象方法

注:以下都是以公有为前提,私有方法只能在类内部调用,不需多讲. 1.对象方法 这种方法都有一个默认参数:self  这代表实例的这个对象 def __init__(self): print("初始化对象") 类是不能直接调用对象方法: class User(object): name = 'zs' def __init__(self): print("初始化对象") User.__init__() 这样调用抛出一个错误:TypeError: __init__() m

selenium grid中的多个线程同步执行

需求:有一个工作流,每一步审批都需要多个领导参与,才能推流程到下一步去 代码思考:多个领导在自己的线程中运行,速度有的快有的慢,如何保证下一步的领导审批时,这个步骤已经激活 如下是代码:思路为:如果这个步骤已激活,则可以进行这个步骤的工作,如果未激活,令他等待一秒,循环等待 for (int i = 0; i < 10000; i++) {     String keyword = (String) ((JavascriptExecutor) driver)       .executeScri