改善C#程序的建议8:避免锁定不恰当的同步对象

简介:

在C#中让线程同步的另一种编码方式就是使用线程锁。所谓线程锁,就是锁住一个资源,使得应用程序只能在此刻有一个线程访问该资源。可以用下面这句不是那么贴切的话来理解线程锁的作用:锁,就是让多线程变成单线程。在C#中,可以将被锁定的资源理解成new出来的普通对象。

既然需要锁定的资源就是一个C#中的对象,我们就该仔细思考,到底什么样的对象能够成为一个锁对象(也叫同步对象)?在选择同步对象的时候,应当始终注意以下几点:

    q同步对象在需要同步的多个线程中是可见的、同一个对象;

    q非静态方法中,静态变量不应作为同步对象;

    q值类型对象不能作为同步对象;

    q避免将字符串作为同步对象。

    q降低同步对象的可见性。

 

第一点,需要锁定的对象在多个线程中是可见的、同一个对象

“可见的”这是显而易见的,如果对象不可见,就不能被锁定。“同一个对象”,这理解起来也很好理解,如果锁定的不是同一个对象,那又如何来同步两个对象呢?可是,不见得我们在这上面不会犯错误。为了阐述本建议,我们先模拟一个必须使用到锁的场景:在遍历一个集合的过程中,同时在另外一个线程中删除集合中的某项。下面的这个例子中,如果没有lock语句,将会抛出异常InvalidOperationException:“集合已修改;可能无法执行枚举”:

复制代码
public   partial   class  FormMain : Form
{
public  FormMain()
{
InitializeComponent();
}

AutoResetEvent autoSet 
=   new  AutoResetEvent( false );
List
< string >  tempList  =   new  List < string > () {  " init0 " " init1 " " init2 "  };

private   void  buttonStartThreads_Click( object  sender, EventArgs e)
{
object  syncObj  =   new   object ();

Thread t1 
=   new  Thread(()  =>
{
// 确保等待t2开始之后才运行下面的代码
autoSet.WaitOne();
lock  (syncObj)
{
foreach  (var item  in  tempList)
{
Thread.Sleep(
1000 );
}
}
});
t1.IsBackground 
=   true ;
t1.Start();

Thread t2 
=   new  Thread(()  =>
{
// 通知t1可以执行代码
autoSet.Set();
// 沉睡1秒是为了确保删除操作在t1的迭代过程中
Thread.Sleep( 1000 );
lock  (syncObj)
{
tempList.RemoveAt(
1 );
}
});
t2.IsBackground 
=   true ;
t2.Start();
}
}
复制代码

这是一个Winform窗体应用程序,我们需要演示的功能在按钮的点击事件中。对象syncObj对于线程t1和t2来说,在CLR中肯定是同一个对象。所以上面的示例运行没有问题。

现在,我们将以上示例重构一下。将实际的工作代码移到一个类型SampleClass中去,该示例要在多个SampleClass实例间操作一个静态字段:

复制代码
private   void  buttonStartThreads_Click( object  sender, EventArgs e)
{
SampleClass sample1 
=   new  SampleClass();
SampleClass sample2 
=   new  SampleClass();
sample1.StartT1();
sample2.StartT2();
}

class  SampleClass
{
public   static  List < string >  TempList  =   new  List < string > () {  " init0 " " init1 " " init2 "  };
static  AutoResetEvent autoSet  =   new  AutoResetEvent( false );
object  syncObj  =   new   object ();

public   void  StartT1()
{
Thread t1 
=   new  Thread(()  =>
{
// 确保等待t2开始之后才运行下面的代码
autoSet.WaitOne();
lock  (syncObj)
{
foreach  (var item  in  TempList)
{
Thread.Sleep(
1000 );
}
}
});
t1.IsBackground 
=   true ;
t1.Start();
}

public   void  StartT2()
{
Thread t2 
=   new  Thread(()  =>
{
// 通知t1可以执行代码
autoSet.Set();
// 沉睡1秒是为了确保删除操作在t1的迭代过程中
Thread.Sleep( 1000 );
lock  (syncObj)
{
TempList.RemoveAt(
1 );
}
});
t2.IsBackground 
=   true ;
t2.Start();
}
}
复制代码

该示例运行起来抛出异常InvalidOperationException:“集合已修改;可能无法执行枚举”。查看类型SampleClass的方法StartT1和StartT2,方法内部锁定的是SampleClass的实例变量syncObject。实例变量意味着每创建一个SampleClass的实例都会生成一个syncObject对象。在本例中,调用者一共创建了两个SampleClass实例,继而分别调用:

sample1.StartT1();
sample2.StartT2();

以上代码锁定的是两个不同的syncObject,这等于完全没有达到两个线程锁定同一个对象的目的。要修正以上的错误,只要将syncObject变成static就可以了。

另外,思考一下lock(this),我们同样不建议在代码中编写这样的代码。如果两个对象的实例分别执行了锁定的代码,实际锁定的也是两个对象,完全不能达到同步的目的。

 

第二个注意事项:非静态方法中,静态变量不应作为同步对象

我们刚说完,要修正第一点中的示例,需要将syncObject变成static。这似乎和本注意事项有矛盾。实际上,第一点中的示例代码仅出于演示的目的。我们强烈建议你不要在实际应用中编写此类代码,在编写多线程代码时,要遵循这样的一个原则:类型的静态方法应当保证线程安全,非静态方法不需实现线程安全。FCL中的绝大部分类,都遵循了这个原则。如果将syncObject变成static,就相当于让非静态方法具备线程安全性,这带来的一个问题是,如果应用程序中该类型存在多个实例,在遇到这个锁的时候,都会产生同步,而这可能不是开发者原先所愿意看到的。

 

第三点:值类型对象不能作为同步对象

值类型在传递另一个线程的时候,会创建一个副本,这相当于每个线程锁定的也是两个对象。故,值类型对象不能作为同步对象。第二点实际也可以归结到第一点中。

 

第四点,锁定字符串是完全没有必要,而且相当危险的

这整个过程看上去和值类型正好相反。字符串在CLR中会被暂存到内存里,如果有两个变量被分配了相同内容的字符串,那么这两个引用会被指向同一块内存。所以,如果有两个地方同时使用了lock(“abc”),那么它们实际锁定的是同一个对象,导致整个应用程序被阻滞。

 

第五点:降低同步对象的可见性

可见范围最广的一种同步对象是typeof(SampleClass)。typeof方法所返回的结果,也就是类型的type,是SampleClass的所有实例所共有的,即:所有实例的type都指向typeof方法的结果。这样一来,如果我们lock(typeof(SampeClass)),当前应用程序中的所有SampleClass的实例的线程,将会全部被同步。这样编码是完全没有必要的,这样的同步对象太开放了。

另外,同步对象一般来说,也不应该是一个公共变量或属性。在FCL的早期版本中,一些常用的集合类型,如ArrayList,提供了公共属性SyncRoot,让我们锁定以便进行一些线程安全的操作。所以你一定会觉得我们刚才的结论不正确。其实不然,ArrayList的操作,大部分的应用场景不涉及到多线程同步,所以它的方法更多的是单线程应用场景。线程同步是一个非常耗时(也就是低效)的操作。若ArrayList的所有非静态方法都要考虑线程安全,那么ArrayList完全可以将这个SyncRoot变成静态私有。现在它将SyncRoot变为公开的,是让调用者自己去决定操作是否需要线程安全。在我们自己编写的大部分代码中,除非也有这样的要求,否则就应该始终考虑降低同步对象的可见性,将我们的同步对象藏起来,只开放给自己或自己的子类就够了(需要开放给子类的情况其实也不多见)。



本文转自最课程陆敏技博客园博客,原文链接:http://www.cnblogs.com/luminji/archive/2011/05/09/2040563.html,如需转载请自行联系原作者

相关文章
|
4月前
|
存储 安全 Java
程序与技术分享:C#值类型和引用类型的区别
程序与技术分享:C#值类型和引用类型的区别
34 0
|
20天前
|
C# 容器
C#中的命名空间与程序集管理
在C#编程中,`命名空间`和`程序集`是组织代码的关键概念,有助于提高代码的可维护性和复用性。本文从基础入手,详细解释了命名空间的逻辑组织方式及其基本语法,展示了如何使用`using`指令访问其他命名空间中的类型,并提供了常见问题的解决方案。接着介绍了程序集这一.NET框架的基本单位,包括其创建、引用及高级特性如强名称和延迟加载等。通过具体示例,展示了如何创建和使用自定义程序集,并提出了针对版本不匹配和性能问题的有效策略。理解并善用这些概念,能显著提升开发效率和代码质量。
33 4
|
26天前
|
Linux C# 开发者
Uno Platform 驱动的跨平台应用开发:从零开始的全方位资源指南与定制化学习路径规划,助您轻松上手并精通 C# 与 XAML 编程技巧,打造高效多端一致用户体验的移动与桌面应用程序
【9月更文挑战第8天】Uno Platform 的社区资源与学习路径推荐旨在为初学者和开发者提供全面指南,涵盖官方文档、GitHub 仓库及社区支持,助您掌握使用 C# 和 XAML 创建跨平台原生 UI 的技能。从官网入门教程到进阶技巧,再到活跃社区如 Discord,本指南带领您逐步深入了解 Uno Platform,并提供实用示例代码,帮助您在 Windows、iOS、Android、macOS、Linux 和 WebAssembly 等平台上高效开发。建议先熟悉 C# 和 XAML 基础,然后实践官方教程,研究 GitHub 示例项目,并积极参与社区讨论,不断提升技能。
36 2
|
1月前
|
C# 数据安全/隐私保护
C# 一分钟浅谈:类与对象的概念理解
【9月更文挑战第2天】本文从零开始详细介绍了C#中的类与对象概念。类作为一种自定义数据类型,定义了对象的属性和方法;对象则是类的实例,拥有独立的状态。通过具体代码示例,如定义 `Person` 类及其实例化过程,帮助读者更好地理解和应用这两个核心概念。此外,还总结了常见的问题及解决方法,为编写高质量的面向对象程序奠定基础。
16 2
|
2月前
|
缓存 NoSQL Redis
【Azure Redis 缓存】C#程序是否有对应的方式来优化并缩短由于 Redis 维护造成的不可访问的时间
【Azure Redis 缓存】C#程序是否有对应的方式来优化并缩短由于 Redis 维护造成的不可访问的时间
|
2月前
|
安全 C# 开发者
【C# 多线程编程陷阱揭秘】:小心!那些让你的程序瞬间崩溃的多线程数据同步异常问题,看完这篇你就能轻松应对!
【8月更文挑战第18天】多线程编程对现代软件开发至关重要,特别是在追求高性能和响应性方面。然而,它也带来了数据同步异常等挑战。本文通过一个简单的计数器示例展示了当多个线程无序地访问共享资源时可能出现的问题,并介绍了如何使用 `lock` 语句来确保线程安全。此外,还提到了其他同步工具如 `Monitor` 和 `Semaphore`,帮助开发者实现更高效的数据同步策略,以达到既保证数据一致性又维持良好性能的目标。
32 0
|
2月前
|
C#
WPF/C#:程序关闭的三种模式
WPF/C#:程序关闭的三种模式
33 0
|
4月前
|
开发框架 .NET 编译器
程序与技术分享:C#基础知识梳理系列三:C#类成员:常量、字段、属性
程序与技术分享:C#基础知识梳理系列三:C#类成员:常量、字段、属性
30 2
|
4月前
|
数据采集 XML 存储
技术经验分享:C#构造蜘蛛爬虫程序
技术经验分享:C#构造蜘蛛爬虫程序
28 0