[C#]Thread Safe Dictionary in .NET 2.0
using System.Collections.Generic;
using System.Threading;
namespace CSharpUtilHelpV2
{
/// <summary>
/// 基于.NET 2.0的对于Dictionary线程安全操作工具类
/// 说明
/// 默认读锁超时1000毫秒
/// 默认写锁超时1000毫秒
/// .NET 4.0 可以使用ConcurrentDictionary来实现。
/// </summary>
/// <typeparam name="T">泛型</typeparam>
public class ThreadSafeDictionaryV2<T>
{
/*参考资料
* 1.
* 使用Monitor或Mutex进行同步控制的问题:由于独占访问模型不允许任何形式的并发访问,这样的效率总是不太高。
* 许多时候,应用程序在访问资源时是进行读操作,写操作相对较少。为解决这一问题,
* C#提供了System.Threading.ReaderWriterLock类以适应多用户读/单用户写的场景。该类可实现以下功能:
* 如果资源未被写操作锁定,那么任何线程都可对该资源进行读操作锁定,并且对读操作锁数量没有限制,即多个线程可同时对该资源进行读操作锁定,以读取数据。
* 如果资源未被添加任何读或写操作锁,那么一个且仅有一个线程可对该资源添加写操作锁定,以写入数据。简单的讲就是:读操作锁是共享锁,允许多个线程同时读取数据;
* 写操作锁是独占锁,同一时刻,仅允许一个线程进行写操作。
* 引用链接:http://www.csharpwin.com/dotnetspace/12761r5814.shtml
*
* 2.
* ReaderWriterLock 用于同步对资源的访问。在任一特定时刻,它允许多个线程同时进行读访问,或者允许单个线程进行写访问。
* 在资源不经常发生更改的情况下,ReaderWriterLock 所提供的吞吐量比简单的一次只允许一个线程的锁(如 Monitor)更高。
* 在多数访问为读访问,而写访问频率较低、持续时间也比较短的情况下,ReaderWriterLock 的性能最好。
* 多个读线程与单个写线程交替进行操作,所以读线程和写线程都不会长时间阻止。
* 大多数在 ReaderWriterLock 上获取锁的方法都采用超时值。使用超时可以避免应用程序中出现死锁。
* 如果不使用超时,这两个线程将出现死锁。
* 引用链接:http://msdn.microsoft.com/zh-cn/library/system.threading.readerwriterlock(v=vs.80).aspx
*
* 3.其他链接
* http://tinythreadsafecache.codeplex.com/SourceControl/latest#TinyThreadSafeCache.cs
* http://www.grumpydev.com/2010/02/25/thread-safe-dictionarytkeytvalue/
* http://stackoverflow.com/questions/157933/whats-the-best-way-of-implementing-a-thread-safe-dictionary
* http://stackoverflow.com/questions/15095817/adding-to-a-generic-dictionary-causes-indexoutofrangeexception
*/
ReaderWriterLock rwlock = new ReaderWriterLock();//声明读写锁 .NET 3.5+ 推荐用ReaderWriterLockSlim
Dictionary<string, T> dic = new Dictionary<string, T>();//Dictionary
static int readerTimeout = 1000;//默认读锁超时1000毫秒
static int writerTimeout = 1000;//默认写锁超时1000毫秒
/// <summary>
/// 默认构造函数
/// </summary>
public ThreadSafeDictionaryV2()
{
}
/// <summary>
/// 带参构造函数
/// </summary>
/// <param name="_readerTimeout">读锁超时设置【单位毫秒】</param>
/// <param name="_writerTimeout">写锁超时设置【单位毫秒】</param>
public ThreadSafeDictionaryV2(int _readerTimeout, int _writerTimeout)
{
readerTimeout = _readerTimeout;
writerTimeout = _writerTimeout;
}
/// <summary>
/// This【线程安全】
/// </summary>
/// <param name="key">键</param>
/// <returns>值</returns>
public T this[string key]
{
get
{
rwlock.AcquireReaderLock(readerTimeout);
try
{
return dic[key];
}
finally
{
rwlock.ReleaseReaderLock();
}
}
set
{
rwlock.AcquireWriterLock(writerTimeout);
try
{
dic[key] = value;
}
finally
{
rwlock.ReleaseWriterLock();
}
}
}
/// <summary>
/// Add【线程安全】
/// 默认超时1000毫秒
/// </summary>
/// <param name="key">键</param>
/// <param name="val">值</param>
public void Add(string key, T val)
{
Add(key, val, writerTimeout);
}
/// <summary>
/// Add【线程安全】
/// </summary>
/// <param name="key">键</param>
/// <param name="val">值</param>
/// <param name="timeout">超时设置【毫秒】</param>
public void Add(string key, T val, int timeout)
{
rwlock.AcquireWriterLock(timeout);
try
{
dic[key] = val;
}
finally
{
rwlock.ReleaseWriterLock();
}
}
/// <summary>
/// Get【线程安全】
/// 默认超时1000毫秒
/// </summary>
/// <param name="key">键</param>
/// <returns>值</returns>
public T Get(string key)
{
return Get(key, readerTimeout);
}
/// <summary>
/// Get【线程安全】
/// </summary>
/// <param name="key">键</param>
/// <param name="timeout">超时设置【毫秒】</param>
/// <returns>值</returns>
public T Get(string key, int timeout)
{
rwlock.AcquireReaderLock(timeout);
try
{
T val;
dic.TryGetValue(key, out val);
return val;
}
finally
{
rwlock.ReleaseReaderLock();
}
}
/// <summary>
/// Remove【线程安全】
/// 默认超时1000毫秒
/// </summary>
/// <param name="key">键</param>
public void Remove(string key)
{
Remove(key, writerTimeout);
}
/// <summary>
/// Remove【线程安全】
/// </summary>
/// <param name="key">键</param>
/// <param name="timeout">超时设置【毫秒】</param>
public void Remove(string key, int timeout)
{
rwlock.AcquireWriterLock(timeout);
try
{
dic.Remove(key);
}
finally
{
rwlock.ReleaseWriterLock();
}
}
/// <summary>
/// Clear【线程安全】
/// 默认超时1000毫秒
/// </summary>
public void Clear()
{
Clear(writerTimeout);
}
/// <summary>
/// Clear【线程安全】
/// </summary>
/// <param name="timeout">超时设置【毫秒】</param>
public void Clear(int timeout)
{
rwlock.AcquireWriterLock(timeout);
try
{
dic.Clear();
}
finally
{
rwlock.ReleaseWriterLock();
}
}
/// <summary>
/// ContainsKey【线程安全】
/// 默认超时1000毫秒
/// </summary>
/// <param name="key">键</param>
/// <returns>是否包含</returns>
public bool ContainsKey(string key)
{
return ContainsKey(key, readerTimeout);
}
/// <summary>
/// ContainsKey【线程安全】
/// </summary>
/// <param name="key">键</param>
/// <param name="timeout">超时设置【毫秒】</param>
/// <returns>是否包含</returns>
public bool ContainsKey(string key, int timeout)
{
rwlock.AcquireReaderLock(timeout);
try
{
return dic.ContainsKey(key);
}
finally
{
rwlock.ReleaseReaderLock();
}
}
/// <summary>
/// Count【线程安全】
/// 默认超时1000毫秒
/// </summary>
/// <returns></returns>
public int Count()
{
return Count(readerTimeout);
}
/// <summary>
/// Count【线程安全】
/// </summary>
/// <param name="timeout">超时设置【毫秒】</param>
/// <returns>Count</returns>
public int Count(int timeout)
{
rwlock.AcquireReaderLock(timeout);
try
{
return dic.Count;
}
finally
{
rwlock.ReleaseReaderLock();
}
}
}
}
.csharpcode, .csharpcode pre
{
font-size: small;
color: black;
font-family: consolas, "Courier New", courier, monospace;
background-color: #ffffff;
/*white-space: pre;*/
}
.csharpcode pre { margin: 0em; }
.csharpcode .rem { color: #008000; }
.csharpcode .kwrd { color: #0000ff; }
.csharpcode .str { color: #006080; }
.csharpcode .op { color: #0000c0; }
.csharpcode .preproc { color: #cc6633; }
.csharpcode .asp { background-color: #ffff00; }
.csharpcode .html { color: #800000; }
.csharpcode .attr { color: #ff0000; }
.csharpcode .alt
{
background-color: #f4f4f4;
width: 100%;
margin: 0em;
}
.csharpcode .lnum { color: #606060; }
[C#]Thread Safe Dictionary in .NET 2.0的更多相关文章
- 【转】php Thread Safe(线程安全)和None Thread Safe(NTS,非 线程安全)之分
Windows版的PHP从版本5.2.1开始有Thread Safe(线程安全)和None Thread Safe(NTS,非线程安全)之分,这两者不同在于何处?到底应该用哪种?这里做一个简单的介绍. ...
- PHP版本VC6与VC9、Thread Safe与None-Thread Safe等的区别
PHP版本VC6与VC9.Thread Safe与None-Thread Safe等的区别 [摘要]PHP 是一种 HTML 内嵌式的语言,是一种在服务器端执行的嵌入HTML文档的脚本语言,在PHP发 ...
- Thread Safe(线程安全)和None Thread Safe(NTS,非线程安全)之分
Windows版的PHP从版本5.2.1开始有Thread Safe(线程安全)和None Thread Safe(NTS,非线程安全)之分,这两者不同在于何处?到底应该用哪种?这里做一个简单的介绍. ...
- PHP的(Thread Safe与Non Thread Safe)
在安装xdebug到时候你会有有TS和NTS版本的选择,在以前还有VC6和VC9的版本.如果你没有根据你目前的服务器的状况选择对应的版本的话,那么xdebug是安装不成功的. 一.如何选择 php5. ...
- PHP版本VC6与VC9/VC11/VC14、Thread Safe与None-Thread Safe等的区别
最近正好在弄一个PHP的程序,在这之前一直没有怎么以接触,发现对PHP版本知识了解不是很清楚,自己看了不少类似的文章,还是感觉不够明确和全面, 网上的结论又都是模棱两可,在此,给出最完整甚至武断的解释 ...
- windows zend_guard+apache no ssl+php no Thread Safe fastcgi模式 环境配置
最近公司要做代码加密,就采用ZEND GUARD 方式加密代码 并进行显示 此文为总结,以备自己以后查看和给需要的同学们参考 采用的php为5.3版本 由于现在加密的更改, 能支持zend guar ...
- PHP关于VC9和VC6以及Thread Safe和Non Thread Safe版本选择的问题
一.如何选择PHP5.3的VC9版本和VC6版本 VC6版本是使用Visual Studio 6编译器编译的,如果你的PHP是用Apache来架设的,那你就选择VC6版本. VC9版本是使用 ...
- PHP关于VC11,VC9,VC6以及Thread Safe和Non Thread Safe版本选择的问题
这里是我在搭建php环境时收集的资料供大家参考: 现在PHP官网上下载PHP安装包都有VC11或VC9的字样,这是什么含义,我们应该下载哪种安装包更好呢?其实PHP官网给出了答案: VC6版本是使用V ...
- PHP版本VC6和VC9、Non Thread Safe和Thread Safe的区别
链接:http://www.cnblogs.com/neve/articles/1863853.html 想更新个PHP的版本,PHP的windows版本已经分离出来了,见http://windows ...
随机推荐
- Linux Shell多进程并发以及并发数控制
1. 基础知识准备 1.1. linux后台进程 Unix是一个多任务系统,允许多用户同时运行多个程序.shell的元字符&提供了在后台运行不需要键盘输入的程序的方法.输入命令后,其后紧跟&a ...
- careercup-树与图 4.2
4.2 给定有向图,设计一个算法,找出两个结点之间是否存在一条路径. 解答 根据题意,给定一个有向图和起点终点,判断从起点开始,是否存在一条路径可以到达终点. 考查的就是图的遍历,从起点开始遍历该图, ...
- 【转】C/C++中的日期和时间 TIME_T与STRUCT TM转换——2013-08-25 16
http://www.cnblogs.com/Wiseman/archive/2005/10/24/260576.html 摘要: 本文从介绍基础概念入手,探讨了在C/C++中对日期和时间操作所用到的 ...
- linux监控程序
IP内部地址: 方法一:ifconfig |grep 'inet addr'|grep '192.168.1' | awk '{print $2}'| tr -d "addr:" ...
- 触摸与手势学习-swift
触摸是一个UITouch对象,该对象放在一个UIEvent中,每个UIEvent包含一个或多个UITouch对象,每个UITouch对象对应一个手指.系统将UIEvent发送到应用程序上,最后应用程序 ...
- 【转载】soapui基于持续集成工具自动化运行的调研姿势
soapui中的testrunner.bat调研姿势,用于自动化测试副标题:soapui基于持续集成工具自动化运行的调研姿势 各位亲爱的同仁们,大家好吗?最近项目在搞持续集成工具,我们的测试用例都是基 ...
- android 中uri.parse()用法
android 中uri.parse()用法 1,调web浏览器 Uri myBlogUri = Uri.parse("http://xxxxx.com"); returnIt = ...
- Ganymed SSH-2 for Java
Ganymed SSH-2 for Java是一个纯Java实现的SHH2库,官网为http://www.ganymed.ethz.ch/ssh2/,最新的更新时间为2006年10月,在用之前,请仔细 ...
- sklearn两种保存模型的方式
作者:卢嘉颖 链接:https://www.zhihu.com/question/27187105/answer/97334347 来源:知乎 著作权归作者所有,转载请联系作者获得授权. 1. pic ...
- C# 反射的简单用法
新建两个项目:类库(Model)和控制台应用程序(ReflectTest). 在[Model]中添加一个类[User]: namespace Model { public class User { p ...