1.缘起:
对于需要进行线程同步的地方,我们经常用的就是.NET内置的lock关键字和ReaderWriterLock类。lock的功能相对简单,因为它不区分读写,也就是说如果都在lock块中,读线程都会阻塞另一个读线程,在很多读远远多于写的应用中,这会极大地折损性能。所以我们也经常需要使用读写分离的锁ReaderWriterLock,使用它,我们可以明确的指定是要获取“读”锁还是“写”锁。而且,当前的“读”线程是不会阻塞其它的“读”线程的。
lock的使用非常简洁,而ReaderWriterLock类的使用就要繁琐很多,比如像这样:
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->
privateReaderWriterLockreaderWriterLock=newReaderWriterLock();
publicvoidTest()
{
try
{
this.readerWriterLock.AcquireWriterLock(-1);
//DoSomething
}
finally
{
this.readerWriterLock.ReleaseWriterLock();
}
}
于是,我设计了ESBasic.Threading.Synchronize.SmartRWLocker来简化ReaderWriterLock的使用,使得我们可以像使用lock一样来使用ReaderWriterLock。
2.适用场合:
在大多数使用ReaderWriterLock的地方都可以使用SmartRWLocker来代替,除非你需要使用某些ReaderWriterLock的特殊功能。SmartRWLocker适用于以下场合:
(1)需要使用读写分离的锁。
(2)不需要设置等待锁的超时时间,也就是无限期地等待锁。
(3)不需要升级/降级锁,如将读锁升级为写锁,或将写锁降级为读锁。
3.设计思想与实现
SmartRWLocker的类图如下:
我们看到SmartRWLocker内部就是借助ReaderWriterLock来实现锁的控制的。而SmartRWLocker只有一个Lock方法,参数是一个AccessMode枚举,表示调用者是希望获取读锁还是写锁,另外该方法返回一个LockingObject对象。LockingObject的生命周期很有意思,LockingObject对象产生的时候,就是获取锁的时刻,其被销毁的时候(Dispose方法),就是释放锁的时刻。所以LockingObject对象的生命周期就是占用锁的时间段。
IDisposable接口与using结合起来使用,会使得语法非常简单可读。我们可以这样来简洁地使用SmartRWLocker:
<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->
privateSmartRWLockersmartRWLocker=newSmartRWLocker();
publicvoidTest2()
{
using(this.smartRWLocker.Lock(AccessMode.Write))
{
//DoSomething
}
}
这就非常类似lock的使用方式了。
最后,LastRequireReadTime和LastRequireWriteTime属性记录了最后一次获取读写锁的时间 -- 即从一个侧面记录了我们对目标资源最后一次进行读写的时间。
4. 使用时的注意事项
SmartRWLocker简化了ReaderWriterLock的使用,但是正如有得必有失,它也损失了一些ReaderWriterLock的功能,正如在适用场合中介绍的,使用SmartRWLocker无法设置获取锁的超时时间,也无法升级/降级锁的性质。幸运的是,大多数情况下,我们都用不到这些高级一点的特性,所以,SmartRWLocker还是有它存在的价值的。
如果你的应用需要使用SmartRWLocker不提供的特性,那只有转向使用ReaderWriterLock本身了。这也未必是个坏主意。
使用任何类型的锁的时候,你都需要注意锁的“粒度”的问题,即你的锁要锁住的范围有多大。粒度太大,会降低系统的并发;粒度太细,又会使得编程相当繁琐。所以在设计时需要进行权衡,为你的锁选择一个恰当的粒度是非常重要的。
5.扩展
简易的读写锁SmartRWLocker暂时没有任何扩展。 注:ESBasic源码可到http://esbasic.codeplex.com/下载。
ESBasic讨论:37677395
ESBasic开源前言
分享到:
相关推荐
本人将 zhuweisky博主的博客整理成了PDF文件,以便于脱机浏览,没有经过博主的同意就这么做 实在是不好意思^-^ 现将其资料免费下载 以示对博主的尊重 源博客地址:...
.NET设计规范:约定、惯用法与模式,在.net环境下进行开发设计的经典之作,权威指导.
LiteGo:「迷你」的Android异步并发类库LiteGo是一款基于Java语言的「异步并发类库」,它的核心是一枚「迷你」并发器,它可以自由地设置同一时段的最大「并发」数量,等待「排队」线程数量,还可以设置「排队策略」...
新版根据.NET Framework 3.0和3.5的新特性做了全面更新,主要关注的是直接影响框架可编程能力的设计问题。遵守这些规范对于使用.NET Framework创建高质量的应用程序至关重要。 本书提供配套光盘,内含Designing ...
NET应用架构设计原则、模式与实践.pdf NET应用架构设计原则、模式与实践.pdf
应用系统建立在此框架之上,采用构件式、可复用开发,节省开发成本,加快开发速度,在软件开发上更好的做到多快省。 适合低中高任意开发水平的开发者,可以开发OA、ERP、BPM、CRM、WMS、TMS、MIS、BI、电商平台后台、...
新版根据.NET Framework 3.0和3.5的新特性做了全面更新,主要关注的是直接影响框架可编程能力的设计问题。遵守这些规范对于使用.NET Framework创建高质量的应用程序至关重要。 本书提供配套光盘,内含Designing ...
博文地址:http://blog.csdn.net/csnd_ayo/article/details/72457190
2.2 框架设计的基本原则 10 2.2.1 场景驱动设计的原则 11 2.2.2 低门槛原则 17 2.2.3 自说明对象模型原则 20 2.2.4 分层架构原则 25 2.3 小结 27 第3章 命名规范 28 3.1 大小写约定 29 3.1.1 ...
自己写的ado.net类库,已实现有关数据库操作的各种方法,子类只需提供数据库连接,即可调用,极大了实现了代码复用.这是我针对实际开发中经常需要重复开发数据库操作而提出的解决方案. 注意,该类实现的是关于sqlserver...
这里收集一些著名的 C/C++ 开发库、SDK、类库、可复用类与结构代码 等信息,列举它们的介绍、参考和网站链接,为各位 C/C++ 程序员和爱好者提供检索和查阅类库的方便。 下面收集的 C/C++ 类库介绍整理来源于文章:...
自从.NET推出以来,他已使用.NET帮助很多行业的用户开发了体现其商业理念的软件产品.Xin Chen是.NET和EAI方面的专家,他与Microsoft和Accenture等多家技术领先的公司合作,为它们的客户提供了优秀的解决方案....
这些规范历经.NET框架三个版本的长期开发,凝聚了数千名开发人员的经验和智慧。微软的各开发组正在使用这些规范开发下一代影响世界的软件产品。 第1章 概述 1 1.1 精心设计的框架所具备的品质 2 1.1.1 精心...
此外,还可以创建一个新类库实现软件复用。 • 第十章介绍.NET下的数据库支持,阐述了ADO.NET的基本概念和结构,并通过示例詊细介绍C#下Web Service数据库访问的方法。 • 第十一章将在以前的基础上介绍一个...
Android应用源码开发Demo,主要用于毕业设计学习。
Android应用源码开发Demo,主要用于毕业设计学习。
本书结合设计实例从面向对象的设计中精选出23个设计模式,总结了面向对象设计中最有价值的经验,并且用简洁可复用的形式表达出来。本书分类描述了一组设计良好、表达清楚的软件设计模式,这些模式在实用环境下特别...
泛型技巧系列:用泛型打造可复用的抽象工厂 体验.net2.0的优雅(四):Provider、策略、控制反转和依赖注入 泛型最佳实践 asp.net 2.0下嵌套masterpage页的可视化编辑 C# 2.0与泛型 动态调用对象的属性和方法——...
CRMEB-DT小程序公众号h5商城v4.0.2商业版 美妆H5模版(多端合一) 手册:https://help.crmeb.net/crmeb-v4/1863402 演示:http://demo26.crmeb.net/admin/ ...7.组件化开发,可复用,开发便捷 8.可视化编辑快速装修主页