写点什么

构建更好的线程安全集合

  • 2009-02-26
  • 本文字数:1027 字

    阅读完需:约 3 分钟

大部分线程安全的集合都有一些基础性的缺陷:虽然每个操作都是线程安全的,但是多个操作无法组合起来使用。这意味着一些基本的执行顺序,例如在弹出顶部元素之前检查栈内元素数量会出现潜在的危险。尽管已经有一些 API 设法将某些操作绑定起来(例如.NET 4 的Coordination Data Structures ),但是它们往往会引入丑陋的方法(如TryDequeue)。

.NET 1 里的集合尝试了另一种方式,它们会对外暴露一个SyncRoot 属性,而不是在内部进行锁定。虽然SyncRoot 仍然是同步对象的默认机制,但是.NET 2 已经抛弃了SyncRoot/Wrapper 设计模式

那么该如何创建一个可用的组合式API 呢?Jared Parson 认为集合不应该直接暴露出线程安全的API,所有的方法都应该属于一个临时的对象,而这个对象只有在您锁定集合的时候才被创建出来。这个临时对象是集合的“钥匙”,只有钥匙的持有者才能获取集合内容。

以下示例为 Jared Parsons 的线程安全队列

复制代码
static void Example1(ThreadSafeQueue<int></int> queue) {<br></br> using (var locked = queue.Lock()) {<br></br> if (locked.Count > 0) {<br></br> var first = locked.Dequeue();<br></br> }<br></br> }<br></br>}

名为 locked 的对象本身不是线程安全的,但是开发人员只有在 using 代码块中才能正确执行操作。在遵守了这一简单规则之后,开发块里的所有代码就是线程安全的。Jared 解释道:

与大部分线程安全的设计一样,这些代码还是有被误用的可能:

  1. 在 ILockedQueue 销毁之后却继续使用它。这种做法应该被禁止,用户现有的知识一般足以避免这个问题。此外一些静态检查工具,例如 FxCop,会把这种做法识别为一个错误。我们也可以使用一种更严厉的做法来阻止此类情况出现:添加一个 disposed 标记,并在每个方法中进行检查。
  2. 如果用户在跨越多个 Lock 语句的情况下保留某个值(例如 Count),那么可能会对集合的状况出现错误的判断和假设。
  3. 如果用户没有正确销毁 ILockedQueue,那么这个对象会被永久锁定。幸运的是,对于实现了 IDisposable 的对象,FxCop 同样会将这种做法识别为一个错误——尽管这不是一个万分稳妥的机制。
  4. 无法确定用户是否会长期持有 ILockedQueue 对象。虽然 IDisposable 一般包含着“短期”的意味,但是这并不能做出完美的保证。
  5. ILockedQueue 并不是线程安全的。虽然一般情况下用户不会把 IDisposable 对象交给多个线程使用,但这也是必须考虑到的情况之一。

查看英文原文: Building a Better Thread-safe Collection

2009-02-26 06:281966
用户头像

发布了 157 篇内容, 共 62.2 次阅读, 收获喜欢 6 次。

关注

评论

发布
暂无评论
发现更多内容

分享精选文章合集 - 2025-10-27

Y11

求职 找工作 招聘 应届生

AWS CloudTrail 可观测最佳实践

观测云

aws cloudtrail

C#线性查找算法

追逐时光者

C#

【第三期】USM上海线下工作坊

ShineScrum

敏捷活动

传帮带 人才梯队建设经验总结(15)

万里无云万里天

人才培养 工业 工厂运维

跨区域协同,破局AI落地痛点!「AI共创 三生万物」司马阅2025企业AI落地应用峰会苏州站完美收官

司马阅

AI赋能“一带一路”:司马阅入选《“一带一路”人工智能应用场景案例集(2025)》

司马阅

不仅仅是代码助手:用 Plugins 将 Claude Code 打造成你的专属工具链 - 概念篇(1/4)

Robin Min

LLM plugins vibe coding Claude-Code Coding Agent

C#/.NET/.NET Core技术前沿周刊 | 第 59 期(2025年10.20-10.26)

追逐时光者

C# .net

PDF/epub一键转思维导图,AI帮你10分钟搞定全书精华

阿星AI工作室

产品 AI 工具 电子书

浅谈 Agent 开发工具链演进历程

阿里巴巴云原生

阿里云 云原生 agent

研发排查问题的利器:一款方法调用栈跟踪工具

京东科技开发者

常用Web 实时通信技术:原理+选型,一篇通关

京东科技开发者

云栖实录 | 实时计算 Flink 全新升级 - 全栈流处理平台助力实时智能

阿里云大数据AI技术

flink 阿里云 AI 实时计算 Fluss

Kingbase与ETL:如何实现金融级数据库的安全数据同步

谷云科技RestCloud

数据处理 数据同步 ETL KingBase 数据集成平台

本地盘的价格,云磁盘的弹性:百度智能云弹性临时盘来了

Baidu AICLOUD

块存储 #云存储 云磁盘

什么是SD-WAN?它的优势有哪些?如何搭建SD-WAN?

光联世纪

SD-WAN

JDD Oxygen智能零售论坛 | 《大模型时代的广告营销变革与实践》

京东科技开发者

技术实践:在基于 RISC-V 的 ESP32 上运行 MQTT over QUIC

EMQ映云科技

告别“信息孤岛”,BeeWorks 如何为制造业数智化转型搭好“底座”?

BeeWorks

即时通讯 IM 私有化部署

企业选择IM即时通讯软件,为何更注重私有化部署?

BeeWorks

即时通讯 IM 私有化部署

采购管理系统

深圳亥时科技

压缩指针:64位系统下,Java虚拟机是如何“偷”回4字节内存的?

poemyang

Java虚拟机 java对象

Apache RocketMQ × AI:面向 Multi-Agent 的事件驱动架构

阿里巴巴云原生

阿里云 云原生 Apache RocketMQ

解决 Nginx 路径前缀问题:从 /serviceA 到 /special/serviceA 的平滑迁移

玄兴梦影

nginx 代理 nginx反向代理 Nginx 代理

瑞幸咖啡成立八周年,向原产地捐赠8所“博爱校医室”

Lily

发布会回顾|袋鼠云发布多模态数据中台,重构AI时代的数据底座

袋鼠云数栈

数据中台 发布会 多模态数据 袋鼠云 数栈

ESP32 + MCP over MQTT:从 0 到 1 打造情感陪伴智能体之「硬件设备能力封装」

EMQ映云科技

ESP32 + MCP over MQTT:通过大模型控制智能硬件设备

EMQ映云科技

橱窗用LED显示屏,店铺“亮”起来

Dylan

品牌 LED LED display LED显示屏 LED屏幕

构建更好的线程安全集合_.NET_Jonathan Allen_InfoQ精选文章