惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
H
Hacker News: Front Page
P
Palo Alto Networks Blog
T
ThreatConnect
Apple Machine Learning Research
Apple Machine Learning Research
博客园_首页
T
True Tiger Recordings
P
Privacy & Cybersecurity Law Blog
B
Blog
IT之家
IT之家
Last Week in AI
Last Week in AI
F
Full Disclosure
Hacker News: Ask HN
Hacker News: Ask HN
C
Comments on: Blog
Microsoft Azure Blog
Microsoft Azure Blog
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
博客园 - 【当耐特】
N
News and Events Feed by Topic
NISL@THU
NISL@THU
腾讯CDC
雷峰网
雷峰网
Security Latest
Security Latest
李成银的技术随笔
M
Microsoft Research Blog - Microsoft Research
L
LangChain Blog
L
Lohrmann on Cybersecurity
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
C
Check Point Blog
Y
Y Combinator Blog
Recent Announcements
Recent Announcements
博客园 - Franky
N
News | PayPal Newsroom
V
V2EX
A
About on SuperTechFans
The Register - Security
The Register - Security
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Google Online Security Blog
Google Online Security Blog
MyScale Blog
MyScale Blog
Cisco Talos Blog
Cisco Talos Blog
Vercel News
Vercel News
WordPress大学
WordPress大学
C
Cyber Attacks, Cyber Crime and Cyber Security
The Hacker News
The Hacker News
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
爱范儿
爱范儿
A
Arctic Wolf
L
LINUX DO - 最新话题
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More

博客园 - 水手paul2008

MySQL 数据库归档日志相关 SQL Server 数据库给第三方用户开权限,限制可见内容。 visual svn server 迁移 windbg preview调试winform程序卡死 JavaScript的数字运算不准的问题 asp.net core 大并发设置 SQLServer数据库优化 windbg 调试 winform程序(二车间机台程序) IIS日志分析工具 .Net的三个Timer内Interval是不精确的 IE 碰到<不能执行已释放 Script 的代码> IIS应用程序池设置若干 关于thinkpad t440s的声卡失真问题,竟然带来对旧电脑的惊喜 ArcMap10.4 遇到严重的应用程序错误,无法继续 给汽车换牌照灯的故事 .Net 混淆器ConfuserEx 车辆年检、强险、车船税的关系 .Net下几种ORM Windows 刷新图标命令
.net List.ForEach内部Async/Await 异步的问题
水手paul2008 · 2023-10-07 · via 博客园 - 水手paul2008

在开发异步的WebApi时,发现一个线程并发错误,起初以为SqlSugar问题。后来,经查是自己写的代码问题。但此问题,不遇到,不会知晓,只有异步才会出现此问题。

这种问题,也发生在JavaScript下,代码逻辑是一样的。参看这篇:https://advancedweb.hu/how-to-use-async-functions-with-array-foreach-in-javascript/

一般同步模式List.ForEach会对每个元素同步执行,上一个执行完成才执行下一个。

List.ForEach(x=>{
  Do(x);
});

但是,ForEach内部执行异步方法时逻辑并非想象的那样等待,它不等待异步方法执行完成,就会进入下个一元素。这会造成多线程相关错误。

List.ForEach(async x=>{
  await Do(x);//此处不等待
});

这时,可以用foreach等此模式会对List元素一个一个同步执行。

foreach(var x in List)
{
  await Do(x);
} 

针对此问题,我做了个样例,此样例是.Net 7控制台程序。如下:

 1 /*第一种模式 ForEach内部,每个循环,不等待上一步完成,一起执行*/
 2 /*
 3  * Expected output:
 4 Item:1 Before. Thread:1
 5 Item:2 Before. Thread:1
 6 Item:3 Before. Thread:1
 7 Finish. Thread:1
 8 Do:3 Thread:9
 9 Item:3 After. Thread:9
10 Do:2 Thread:9
11 Item:2 After. Thread:9
12 Do:1 Thread:9
13 Item:1 After. Thread:9
14 */
15 //new int[] { 1, 2, 3 }.ToList().ForEach(async x =>
16 //{
17 //    Console.WriteLine($"Item:{x} Before. Thread:{Thread.CurrentThread.ManagedThreadId}");
18 //    await A.Do(x);
19 //    Console.WriteLine($"Item:{x} After. Thread:{Thread.CurrentThread.ManagedThreadId}");
20 //});
21 
22 /*第二种模式 foreach内部,每个循环,等待上一步完成,同步执行*/
23 /*
24  * Expected output: 
25 Item:1 Before. Thread:1
26 Do:1 Thread:4
27 Item:1 After. Thread:4
28 Item:2 Before. Thread:4
29 Do:2 Thread:4
30 Item:2 After. Thread:4
31 Item:3 Before. Thread:4
32 Do:3 Thread:4
33 Item:3 After. Thread:4
34 Finish. Thread:4
35  */
36 foreach (var x in new int[] { 1, 2, 3 })
37 {
38     Console.WriteLine($"Item:{x} Before. Thread:{Thread.CurrentThread.ManagedThreadId}");
39     await A.Do(x);
40     Console.WriteLine($"Item:{x} After. Thread:{Thread.CurrentThread.ManagedThreadId}");
41 
42 }
43 
44 Console.WriteLine($"Finish. Thread:{Thread.CurrentThread.ManagedThreadId}");
45 Console.ReadKey();
46 
47 class A
48 {
49       
50     public static async Task Do(int a)
51     {
52         await Task.Delay((10 - a) * 1000);
53         Console.WriteLine($"Do:{a} Thread:{Thread.CurrentThread.ManagedThreadId}");
54     } 
55 }

若使用List.ForEach习惯了,可以使用.Net扩展功能,添加如下方法。但我觉得,不如用foreach(var x in list) 直观:为了避免多元素同时异步特意使用foreach。

public static async Task ForEachAsync<T>(this IEnumerable<T> source, Func<T, Task> action)
    {
        foreach (var value in source)
        {
            await action(value);
        }
    }