当前位置: 首页 > news >正文

c# 中Parallel.ForEach 对其中一个变量进行赋值 引发报错

在 C# 中使用 Parallel.ForEach 方法时,如果你尝试在并行循环中对共享变量进行赋值,很可能会遇到线程安全问题或竞争条件(race conditions),这可能导致数据不一致、程序崩溃或其他不可预测的行为。

问题描述
假设你有以下代码:

int sharedVariable = 0;
Parallel.ForEach(someCollection, item =>
{// 假设这里有一些计算int result = ComputeSomething(item);// 尝试更新共享变量sharedVariable = result;
});

在这段代码中,sharedVariable 被多个线程同时访问和修改,这是不安全的。每次一个线程尝试写入 sharedVariable 时,它可能会覆盖其他线程之前的结果,或者由于处理器缓存和内存一致性问题,导致最终的值不正确。

解决方案
使用线程安全的集合或变量:
对于简单的整数或浮点数,可以使用 Interlocked 类来确保线程安全的读写操作。
对于更复杂的类型,可以考虑使用 Concurrent 命名空间下的集合,如 ConcurrentBag, ConcurrentQueue, ConcurrentDictionary<TKey, TValue> 等。
对于简单的累加操作,可以使用 Interlocked.Add 或 Interlocked.Increment。
使用局部变量并最后合并:
在每个线程中计算局部结果,然后在并行循环外部合并这些结果。
例如,使用局部变量并在最后合并:

List<int> localResults = new List<int>();
Parallel.ForEach(someCollection, item =>
{int result = ComputeSomething(item);localResults.Add(result);
});int sharedVariable = localResults.Sum();  // 或者其他合并逻辑

使用自定义的线程安全数据结构:
如果内置的数据结构不满足需求,你可以实现自己的线程安全数据结构。
使用锁:
使用 lock 关键字可以确保只有一个线程在任何给定时间可以访问特定的代码块。但是,锁会降低并行性能,应谨慎使用。
例如,使用锁来保护共享变量:

object lockObj = new object();
int sharedVariable = 0;
Parallel.ForEach(someCollection, item =>
{int result = ComputeSomething(item);lock (lockObj){sharedVariable = result;  // 注意:这仍然只会保留最后一个线程的结果}
});

注意:在上面的锁示例中,即使使用了锁,sharedVariable 仍然只会保留最后一个线程计算的结果,因为每次写入都会覆盖前一次的值。如果你需要累加或其他形式的合并,应考虑使用其他方法,如 Interlocked 或局部变量合并。

总之,处理并行编程中的共享资源时,需要特别小心以确保线程安全。选择正确的同步机制对于程序的正确性和性能至关重要。

http://www.lryc.cn/news/516348.html

相关文章:

  • ElasticSearch备考 -- 整体脉络梳理
  • vue Element Ui Upload 上传 点击一个按钮,选择多个文件后直接上传,使用防抖解决多次上传的问题。
  • 【HF设计模式】05-单例模式
  • 运维人员的Python详细学习路线
  • 软件体系结构与设计模式
  • 安徽省地图arcgis数据美化后mxd文件shp格式下载后内容测评
  • MySQL数据库备份与恢复策略
  • go语言zero框架中教务crm系统的在职继承和离职交接的设计与实践
  • C# 设计模式(结构型模式):桥接模式
  • C# 设计模式(行为型模式):解释器模式
  • 如何 cURL Elasticsearch:进入 Shell
  • 深信服云桌面系统的终端安全准入设置
  • Node.js 模块系统
  • 数据结构知识收集尊享版(迅速了解回顾相关知识)
  • SpringMVC启动与请求处理流程解析
  • C++ 日志库 spdlog 使用教程
  • `http_port_t
  • SpringBoot中实现拦截器和过滤器
  • 不锈钢均温板结合强力粘合技术革新手机内部架构
  • Docker安装使用
  • React 如何进行路由变化监听
  • Unity UGUI使用技巧与经验总结(不定期更新)
  • 中国乡镇界shp全境arcgis格式shp数据乡镇名称下载后内容测评
  • 第 31 章 - 源码篇 - Elasticsearch 写入流程深入分析
  • node.js下载、安装、设置国内镜像源(永久)(Windows11)
  • 小于n的最大数 - 贪心算法 - C++
  • 【顶刊TPAMI 2025】多头编码(MHE)之极限分类 Part 3:算法实现
  • 解决CentOS 8 YUM源更新后报错问题:无法下载AppStream仓库元数据
  • [python3]Excel解析库-openpyxl
  • Docker 远程访问完整配置教程以及核心参数理解