首页 > 编程 > .NET > 正文

.net中线程同步的典型场景和问题剖析

2024-07-10 12:41:18
字体:
来源:转载
供稿:网友
在使用多线程进行编程时,有一些经典的线程同步问题,对于这些问题,.net提供了多种不同的类来解决。除了要考虑场景本身,一个重要的问题是,这些线程是否在同一个应用程序域中运行。如果线程都在同一应用程序域中运行,则可以使用一些所谓“轻量”级的同步类,否则要使用另一些类,而这些类都是对操作系统所提供的同步原语的包装,相对来说更消耗资源。我在这儿介绍一些典型的应用场景和相关的问题。 

多线程争用独占资源
常常有一些资源线程独占的,如果有多个线程同时需要访问这要的资源,就形成了一个争用问题。这类资源有“文件”,“打印机”,“串口”,以及所有非线程安全的类对象(绝大部分类库中的类都是)。典型的代码:
代码如下:
var objLock = new Object();
var thread1 = new Thread(() =>
{
lock (objLock)
{
AccessResource();
}
});
var thread2 = new Thread(() =>
{
lock (objLock)
{
AccessResource();
}
});

上面代码中,lock关键字实际上Monitor类的一个语法糖。任意一个对象(非值类型)上都有一个锁区域,Monitor.Enter方法会尝试锁定该区域,如果锁定成功,线程就拥有该对象,反子,线程将被挂起。对于objLock对象,有以下点需要注意:
不要锁定this
不要锁定Type
不要锁定字符串
不要锁定值类型的对象
对于相同的类,通常都会有很多不同的实例,这样的话,有可能会锁定到多个不同的对象上,从而使锁失效。不要锁定Type的原因有两点,一是生成Type类对象相对比较慢比较占资源,二是Type类型通常是公共的,这样有可能会在程序的多个不同地方会锁定,这实际上是个工程问题,主要是为了防止引入BUG。不要锁定string类,是因数,所有字面值相同的字符串,实际上是共享同一个对象的,所以和Type一样,也可能会无意间被别的代码锁定,这样的Bug将难以排除。不要锁定值类型,因为值类型本身是不可锁定的,为了可以锁定,编译器值将它装箱,而每次装箱实际上都会生成一个不同的对象实例,这样锁定也就没有任何效果了。
上面的代码有效的原因是所有线程都在同一个应用程序中,也就是不涉及进程间的资源争用。如果是多进程间的资源争用,可以使用Mutex类。Mutex类有两种不同用法,匿名互斥体和命名互斥体,命名的互斥体是在整个操作系统范围内共用的,所以可以用于进程间同步。
代码如下:
var mutex = new Mutex(false, "name");
var thread1 = new Thread(() =>
{
try
{
mutex.WaitOne();
AccessResource();
}
finally
{
mutex.ReleaseMutex();
}
});
var thread2 = new Thread(() =>
{
try
{
mutex.WaitOne();
AccessResource();
发表评论 共有条评论
用户名: 密码:
验证码: 匿名发表