.NET 同步与异步 之 线程安全的集合 (十一)
本随笔续接:.NET 同步与异步 之 警惕闭包(十)
无论之前说的锁、原子操作 还是 警惕闭包,都是为安全保驾护航,本篇随笔继续安全方面的主题:线程安全的集合。
先看一下命名空间:System.Collections.Concurrent,常用的类型有(均为泛型):BlockingCollection<T>、ConcurrentBag<T>、ConcurrentDictionary<TKey, TValue>、ConcurrentQueue<T>、ConcurrentStack<T> 。
其中:ConcurrentBag<T> 为无序的集合、ConcurrentDictionary<TKey, TValue> 为词典类型。
ConcurrentQueue<T>、ConcurrentStack<T> 分别为队列 和 栈,而 BlockingCollection<T> 可以看做是 队列 和 栈的进一步封装调用,并提供了阻塞(超时)功能。
本随笔着重说两个类型:BlockingCollection<T> 和 ConcurrentDictionary<TKey, TValue>。
一、BlockingCollection<T>
1、先看一下 MSDN 上的Demo
/// <summary>
/// MSDN Demo
/// BlockingCollection<T>.Add()
/// BlockingCollection<T>.CompleteAdding()
/// BlockingCollection<T>.TryTake()
/// BlockingCollection<T>.IsCompleted
/// </summary>
public void Demo1()
{
// Construct and fill our BlockingCollection
using (BlockingCollection<int> bc = new BlockingCollection<int>())
{
int NUMITEMS = ; for (int i = ; i < NUMITEMS; i++)
{
bc.Add(i);
}
bc.CompleteAdding(); int outerSum = ; // Delegate for consuming the BlockingCollection and adding up all items
Action action = () =>
{
int localItem;
int localSum = ; while (bc.TryTake(out localItem))
{
localSum += localItem;
}
Interlocked.Add(ref outerSum, localSum);
}; // Launch three parallel actions to consume the BlockingCollection
Parallel.Invoke(action, action, action); base.PrintInfo(string.Format("Sum[0..{0}) = {1}, should be {2}", NUMITEMS, outerSum, ((NUMITEMS * (NUMITEMS - )) / )));
base.PrintInfo(string.Format("bc.IsCompleted = {0} (should be true)", bc.IsCompleted)); }
}
MSDN Demo
从demo中看一下 BlockingCollection<T> 的用法
1)Add 方法, 将项添加到集合中。
2)CompleteAdding 方法,标记当前实例不可以再添加任何项。
3)TryTake 方法,如果可以从当前集合移除一个项,则返回true,否则返回False. 如果该集合为空,则此方法立即返回 false。
删除了某个项的顺序取决于用于创建集合的类型 BlockingCollection<T> 实例。 当您创建 BlockingCollection<T> 对象,您可以指定要使用的集合类型(通过构造函数指定)。 例如,可以指定 ConcurrentQueue<T> 先进先出 (FIFO) 行为的对象或 ConcurrentStack<T> 后进先出 (LIFO) 行为的对象。 可以使用任何集合类来实现 IProducerConsumerCollection<T> 接口。 默认集合类型 BlockingCollection<T> 是 ConcurrentQueue<T>。
4)IsCompleted 属性,获取此 BlockingCollection<T> 是否已标记为完成添加(即 调用了 CompleteAdding 方法)并且为空。
2、限制容量
/// <summary>
/// 限制容量
/// </summary>
public void Demo2()
{
BlockingCollection<int> blocking = new BlockingCollection<int>(); Task.Run(() =>
{
for (int i = ; i < ; i++)
{
blocking.Add(i);
PrintInfo($"add:({i})");
} blocking.CompleteAdding();
PrintInfo("CompleteAdding");
}); // 等待先生产数据
Task.Delay().ContinueWith((t) =>
{
while (!blocking.IsCompleted)
{
var n = ;
if (blocking.TryTake(out n))
{
PrintInfo($"TryTake:({n})");
}
} PrintInfo("IsCompleted = true");
}); }
限制容量
调研Add方法的时候,如果集合中的项的数量已经达到上限,则Add方法将会被阻塞。
3、在 BlockingCollection 中使用Stack
/// <summary>
/// 在 BlockingCollection 中使用Stack
/// </summary>
public void Demo3()
{
BlockingCollection<int> blocking = new BlockingCollection<int>(new ConcurrentStack<int>(), ); Task.Run(() =>
{
for (int i = ; i < ; i++)
{
blocking.Add(i);
PrintInfo($"add:({i})");
} blocking.CompleteAdding();
PrintInfo("CompleteAdding");
}); // 等待先生产数据
Task.Delay().ContinueWith((t) =>
{
while (!blocking.IsCompleted)
{
var n = ;
if (blocking.TryTake(out n))
{
PrintInfo($"TryTake:({n})");
}
} PrintInfo("IsCompleted = true");
}); }
在 BlockingCollection 中使用Stack
该Demo和之前的Demo的唯一区别就是:构造函数、指定了 ConcurrentStack<T> 类型为存储容器。
除此之外、BlockingCollection<T> 还是提供了对超时的控制,例如:TryAdd(T, TimeSpan) 、 TryTake(T, TimeSpan) 等数个重载版本。
二、ConcurrentDictionary<TKey, TValue>。
ConcurrentDictionary<TKey, TValue> 类, 已经实现 IDictionary<TKey, TValue> 接口。也就是说 它也实现了Dictionary 类型的基础功能。
此外, ConcurrentDictionary<TKey, TValue> 提供了几种方法中添加或更新键/值对在字典中下, 如表中所述。
| 任务 | 使用此方法 | 用法说明 | 
|---|---|---|
| 如果它尚不在字典中存在向字典中,添加新的密钥 | 如果当前不在字典中存在该键,此方法将添加指定的键/值对。 该方法返回 true 或 false具体取决于是否已添加新对。 | |
| 如果该注册表项具有特定值,更新为现有键在字典中值 | 此方法检查是否密钥具有指定的值,如果它存在,则用新值更新该键。 它相当于CompareExchange 方法,但它的用于字典的元素。 | |
| 无条件地将键/值对存储在字典中,覆盖已存在的键的值 | 索引器的资源库︰dictionary[key] = newValue | |
| 将键/值对添加到字典中,或如果键已存在,更新基于键的现有值的键的值 | AddOrUpdate(TKey, Func<TKey, TValue>, Func<TKey, TValue, TValue>) - 或 - | AddOrUpdate(TKey, Func<TKey, TValue>, Func<TKey, TValue, TValue>) 接受的键和两个委托。 如果键在字典; 中不存在,则使用第一个委托它接受键并返回应添加的键的值。 如果该键不存在; 它使用第二个委托它接受的键和其当前值,并返回应为项设置的新值。 AddOrUpdate(TKey, TValue, Func<TKey, TValue, TValue>) 接受键、 值要添加,以及更新委托。 这是与以前的重载中,相同之处在于它不使用委托来添加的键。 | 
| 获取此键在字典中,向字典中添加值并将其返回如果该键不存在的值 | - 或 - | 这些重载提供延迟初始化为键/值对在字典中,添加的值,仅当不存在。 GetOrAdd(TKey, TValue) 采用键不存在要添加的值。 GetOrAdd(TKey, Func<TKey, TValue>) 将一个委托,可将生成的值,如果键不存在。 | 
这些操作是原子性操作,而且都是线程安全的。在 ConcurrentDictionary<TKey, TValue> 类中 唯一的例外是 AddOrUpdate 和 GetOrAdd 方法,它们是使用细粒度锁定,以确保线程安全。
ConcurrentDictionary<TKey, TValue> 类 在上述的MSDN文档中 已经介绍的差不多了,不再举例。 当然还要提一句, 该类型不支持阻塞操作。
三、线程安全警告
在命名空间 System.Collections.Concurrent 中的类型,都遵循如下的线程安全规则:线程安全集合本身提供的方法 是线程安全的,但是通过其类型实现的接口的方法 和 扩展方法 却不是线程安全的。
附,Demo : http://files.cnblogs.com/files/08shiyan/ParallelDemo.zip
参见更多:随笔导读:同步与异步
.NET 同步与异步 之 线程安全的集合 (十一)的更多相关文章
- .NET 同步与异步 之 Mutex (十二)
		本随笔续接:.NET 同步与异步 之 线程安全的集合 (十一) 本随笔 及 接下来的两篇随笔,将介绍 .NET 同步与异步系列 的最后一个大块知识点:WaitHandle家族. 抽象基类:WaitHa ... 
- C++11 半同步半异步线程池的实现
		#include <list> #include <mutex> #include <thread> #include <condition_variable ... 
- python网络编程基础(线程与进程、并行与并发、同步与异步、阻塞与非阻塞、CPU密集型与IO密集型)
		python网络编程基础(线程与进程.并行与并发.同步与异步.阻塞与非阻塞.CPU密集型与IO密集型) 目录 线程与进程 并行与并发 同步与异步 阻塞与非阻塞 CPU密集型与IO密集型 线程与进程 进 ... 
- CIL锁,GIL与线程池的区别,进程池和线程池,同步与异步
		一.GIL锁 什么是GIL? 全局解释器锁,是加在解释器上的互斥锁 GC是python自带的内存管理机制,GC的工作原理:python中的内存管理使用的是应用计数,每个数会被加上一个整型的计数器,表示 ... 
- spring线程池(同步、异步)
		一.spring异步线程池类图 二.简单介绍 2.1. TaskExecutor---Spring异步线程池的接口类,其实质是java.util.concurrent.Executor 以下是官方已经 ... 
- [多线程]线程基础(对象锁、class锁、同步、异步)
		synchronized.volatile.ReentrantLock.concurrent 线程安全:当多个线程访问某一个类(对象或方法)时,这个类始终都能表现出正确的行为,那么这个类(对象或方法) ... 
- Python线程,进程,携程,I/O同步,异步
		只有本人能看懂的-Python线程,进程,携程,I/O同步,异步 举个栗子: 我想get三个url,先用普通的for循环 import requests from multiprocessing im ... 
- java 线程之对象的同步和异步
		一.多线程环境下的同步与异步 同步:A线程要请求某个资源,但是此资源正在被B线程使用中,因为同步机制存在,A线程请求不到,怎么办,A线程只能等待下去. package com.jalja.org.th ... 
- 使用C++11 开发一个半同步半异步线程池
		摘自:<深入应用C++11>第九章 实际中,主要有两种方法处理大量的并发任务,一种是一个请求由系统产生一个相应的处理请求的线程(一对一) 另外一种是系统预先生成一些用于处理请求的进程,当请 ... 
随机推荐
- Unable to load configuration. - action - file:/F:/apache-tomcat-8.0.30/webapps/test1Struts2/WEB-INF/classes/struts.xml:11:71
			Unable to load configuration. - action - file:/F:/apache-tomcat-8.0.30/webapps/test1Struts2/WEB-INF/ ... 
- HDU 4687 Boke and Tsukkomi (一般图最大匹配)【带花树】
			<题目链接> 题目大意: 给你n个点和m条边,每条边代表两点具有匹配关系,问你有多少对匹配是冗余的. 解题分析: 所谓不冗余,自然就是这对匹配关系处于最大匹配中,即该匹配关系有意义.那怎样 ... 
- SQL-内连接、外连接(左、右)、交叉连接
			本文测试基于以下两个表,student(左) \ teacher(右),使用数据库MariaDB,图形化界面HeidiSQL. 连接查询的概念:根据两个表或多个表的列之间的关系,从这些表中查询数据,即 ... 
- Python内存管理以及数据类型
			一.内存管理 1.Cpython解释器的垃圾回收机制 什么是垃圾:当一个值身上没有绑定任何变量名(该值的引用计数=0)时,该值就是一个垃圾. Cpython解释器就会自动回收这样的垃圾. #引用计数增 ... 
- POJ1700----Crossing River
			#include<cstdio> #include<iostream> #include<cstring> #include<algorithm> us ... 
- LOJ.2585.[APIO2018]新家(二分 线段树 堆)
			LOJ 洛谷 UOJ BZOJ 四OJ Rank1 hhhha 表示这个b我能装一年→_→ 首先考虑离线,将询问按时间排序.对于每个在\([l,r]\)出现的颜色,拆成在\(l\)加入和\(r+1\) ... 
- OpenCV3.3.0 + CLion + CMake 配置(Mac巨细无敌版)
			目录 开始 完成了 参考链接: 1,cmake编译 http://blog.csdn.net/baidu_36316735/article/details/53168438 2,CLion导入open ... 
- [UVA227][ACM/ICPC WF 1993]Puzzle (恶心模拟)
			各位大佬都好厉害…… 这个ACM/ICPC1993总决赛算黄题%%% 我个人认为至少要绿题. 虽然算法上面不是要求很大 但是操作模拟是真的恶心…… 主要是输入输出的难. 对于ABLR只需要模拟即可 遇 ... 
- 穷举法、for循环、函数、作用域、斐波那契数
			1.穷举法 枚举所有可能性,直到得到正确的答案或者尝试完所有值. 穷举法经常是解决问题的最实用的方法,它实现起来热别容易,并且易于理解. 2.for循环 for语句一般形式如下: for variab ... 
- react-native 热更新react-native-pushy集成遇到的问题
			主要步骤按官方文档实现,这里只记录遇到的一些小坑 官方文档 run-android时NDK报错 前提是NDK已安装并且环境变量已设置 根据报错提示在android/local.properties文件 ... 
