C#异步并行处理的用法

复制代码
await Task.Run(() =>
{
    Parallel.ForEach(_equips, eq =>
    {
        WriteOneEquipFromDb(eq, lineName, jzName, selectedBatchNo);
    });
});

逐层解析

|----|---------------------------|-------------------|
| 层级 | 组件 | 作用 |
| 1 | `await` | 等待异步操作完成,不阻塞UI线程 |
| 2 | `Task.Run()` | 在线程池线程上执行委托 |
| 3 | `Parallel.ForEach()` | 并行遍历集合,多线程同时处理 |
| 4 | `WriteOneEquipFromDb()` | 实际业务方法(读写PLC+数据库) |

执行流程图

主线程(UI)

await Task.Run() → 创建新线程(线程A)

Parallel.ForEach()

├─ 线程1: WriteOneEquipFromDb(_equips[0])

├─ 线程2: WriteOneEquipFromDb(_equips[1])

├─ 线程3: WriteOneEquipFromDb(_equips[2])

└─ ... 同时执行多个设备

全部完成后返回

继续执行后续代码(OutputInfomation等)

关键特性

1. 双重并行机制

复制代码
// 第一层:Task.Run 把任务放到线程池
Task.Run(() => { ... })  // 避免阻塞UI

// 第二层:Parallel.ForEach 内部再并行
Parallel.ForEach(...)     // 多个设备同时处理

2. 线程安全考虑

WriteOneEquipFromDb 内部使用了 Invoke 更新UI:

复制代码
// 这是正确的做法,因为Parallel.ForEach的线程不是UI线程
Invoke(new Action(() => OutputInfomation(...)));

潜在问题与风险

问题1:PLC连接并发冲突

复制代码
// 危险!多个线程同时连接不同PLC(IP不同,通常安全)
// 但如果 _equips 中有相同IP的设备,会冲突!

private void WriteOneEquipFromDb(Equip eq, ...)
{
    var plc = new SvMitsubishiMC3E();
    plc.Connect(eq.IP, port);  // 每个线程独立连接,通常安全
}

问题2:数据库连接池压力

复制代码
// 每个设备都创建新的数据库连接(通过 _dbHelper)
// 如果设备数量多(>10),可能耗尽连接池

问题3:异常处理不完善

复制代码
// Parallel.ForEach 中一个线程异常,其他线程继续
// 但异常会被聚合抛出,需要正确处理

改进建议

方案A:限制并行度(推荐)

复制代码
await Task.Run(() =>
{
    var options = new ParallelOptions { MaxDegreeOfParallelism = 3 }; // 最多3个并发
    Parallel.ForEach(_equips, options, eq =>
    {
        WriteOneEquipFromDb(eq, lineName, jzName, selectedBatchNo);
    });
});

方案B:使用 async/await 替代 Parallel

复制代码
// 更可控的并发
await Task.Run(async () =>
{
    var tasks = _equips.Select(eq => 
        Task.Run(() => WriteOneEquipFromDb(eq, lineName, jzName, selectedBatchNo))
    );
    await Task.WhenAll(tasks); // 等待全部完成
});

方案C:顺序执行(最安全,但最慢)

复制代码
await Task.Run(() =>
{
    foreach (var eq in _equips) // 不用Parallel
    {
        WriteOneEquipFromDb(eq, lineName, jzName, selectedBatchNo);
    }
});

|----------------|----------------------------------|
| 场景 | 建议方案 |
| 设备少(<5台),不同IP | 当前代码可用 |
| 设备多(>5台),不同IP | 限制 MaxDegreeOfParallelism 为3-4 |
| 设备IP可能重复 | 必须改为顺序执行 |
| 需要精确进度反馈 | 使用 Task.WhenAll + 单独任务跟踪 |

相关推荐
weixin_520649872 小时前
Winform创建与启动
开发语言·c#
武藤一雄3 小时前
WPF中逻辑树(Logical Tree)与可视化树(Visual Tree)到底是什么
microsoft·c#·.net·wpf·.netcore
周杰伦fans3 小时前
深入 C# 匿名类型:从 `new { Ask = ask }` 说起
开发语言·c#
不会编程的懒洋洋3 小时前
C# IDisposable 和 using
开发语言·笔记·机器学习·c#·.net·visual studio·c#基础
家有娇妻张兔兔3 小时前
Apache POI 导出 Word 踩坑实录:Word 分栏为什么做不好左右平铺
c#·word·apache·poi·分栏
唐青枫3 小时前
C#.NET MediatR 深入解析:进程内消息分发、CQRS、通知事件与管道行为实战
c#·.net
njsgcs13 小时前
拆分多实体到装配体 solidworks c#
c#
何以解忧唯有撸码14 小时前
C# 视频录制监控系统
c#·winform