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

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.kler.cn/a/470906.html

相关文章:

  • 数据结构与算法之二叉树: LeetCode 107. 二叉树的层序遍历 II (Ts版)
  • Clisoft SOS与CAD系统集成
  • 学习threejs,导入assimp assimp2json格式的模型
  • 使用 Python结合ffmpeg 实现单线程和多线程推流
  • C++ static关键字(八股总结)
  • 弥散张量分析开源软件 DSI Studio 简体中文汉化版可以下载了
  • 计算机网络•自顶向下方法:多址接入协议
  • 【AI数学基础】线性代数:向量空间
  • reactor的Hooks.enableAutomaticContextPropagation();不生效解决方案
  • 基于32单片机的智能语音家居
  • pytest日志显示
  • gesp(C++一级)(18)洛谷:B4063:[GESP202412 一级] 奇数和偶数
  • 某制造集团灯塔工厂解决方案(36页PPT)
  • 安装vue脚手架出现的一系列问题
  • 计算机网络——网络层—路由算法和路由协议
  • 感知器的那些事
  • springboot适配mybatis+guassdb与Mysql兼容性问题处理
  • 升级 Spring Boot 3 配置讲解 —— Spring Boot 3 核心源码专讲
  • 如何在 Ubuntu 22.04 上安装 Nagios 服务器教程
  • Flutter:打包apk,安卓版本更新(二)
  • 使用Python构建远程医疗平台:从零开始的实现指南
  • 【错误记录】HarmonyOS 编译报错 ( DevEco Studio 开发环境 与 API 版本 与 HarmonyOS 版本 的配套关系 )
  • 君正T41交叉编译ffmpeg、opencv并做h264软解,利用君正SDK做h264硬件编码
  • Angular由一个bug说起之十三:Cross Origin
  • C++二十三种设计模式之外观模式
  • Nginx不使用域名如何配置证书