0

0

c# Task.Delay(-1) 和 Task.WaitAll(tasks) 有什么效果

畫卷琴夢

畫卷琴夢

发布时间:2026-01-03 08:04:29

|

234人浏览过

|

来源于php中文网

原创

Task.Delay(-1) 返回永不完成的Task,等效于Timeout.Infinite,需配合CancellationToken使用;Task.WaitAll同步阻塞线程,禁用在UI/请求线程;二者混用将导致永久冻结,应改用await Task.WhenAll或TaskCompletionSource。

c# task.delay(-1) 和 task.waitall(tasks) 有什么效果

Task.Delay(-1) 会无限等待,但不是“挂起线程”,而是返回一个永不完成的 Task

它等效于 Task.Delay(Timeout.Infinite)(即 int.MaxValue 毫秒),但语义更明确:你主动要求“永远别完成”。这不是错误,也不会抛异常,而是一个合法、可 await 的“死任务”。
常见误用是把它当“暂停直到手动唤醒”——但它没有取消机制,除非你传入 CancellationToken

  • ✅ 正确用法:await Task.Delay(-1, cancellationToken),配合外部取消触发退出
  • ❌ 错误假设:Task.Delay(-1).Wait().Result —— 这会永久阻塞当前线程(尤其在 UI 线程中 = 界面卡死)
  • ⚠️ 注意:.NET 6+ 中 Task.Delay(-1) 已被标记为 Obsolete,编译器会警告,推荐改用 Task.Delay(Timeout.InfiniteTimeSpan)

Task.WaitAll(tasks) 是同步阻塞,不是异步等待

它会让**当前线程停住不动**,直到所有传入的 Task 对象都进入 RanToCompletionFaultedCanceled 状态。它不关心这些任务是不是 async 方法启动的,只看底层 Task 实例的状态。

  • ✅ 合理场景:后台服务的 Main 方法中协调多个 CPU 密集型 Task.Run,且你确定不会阻塞 UI 或请求线程
  • ❌ 绝对禁止:在 WinForms/WPF/ASP.NET Core 请求线程(如控制器 action)里调用 —— WaitAll 会锁死线程池线程,引发严重吞吐下降甚至死锁
  • ⚠️ 异常处理必须 catch AggregateException,因为任意一个 task 抛异常都会被打包抛出

两者混用是典型反模式:比如 Task.WaitAll(Task.Delay(-1))

这等于让当前线程永远卡住 —— 因为 Task.Delay(-1) 永不完成,Task.WaitAll 就永远等下去。没有任何超时、无法中断(除非线程被强行 Abort,但 .NET Core+ 已禁用)、不可调试。

ModelScope
ModelScope

魔搭开源模型社区旨在打造下一代开源的模型即服务共享平台

下载
  • ❌ 错误示例:
    var t = Task.Delay(-1);
    Task.WaitAll(t); // 主线程在此处彻底冻结
  • ✅ 替代方案:用 await Task.WhenAll(t) + CancellationToken 控制生命周期;或改用 TaskCompletionSource 手动控制完成时机
  • ? 关键区别WhenAll 返回 Task 可 await,不阻塞;WaitAll 是 void 方法,强制同步等待

真正需要“等一个不确定时间”的时候,该用什么?

多数情况下,你要的不是“无限延迟”,而是“等某个条件满足”或“等外部信号”。直接硬写 Task.Delay(-1) 或死等 WaitAll,说明设计上漏掉了取消、超时或事件驱动机制。

  • TaskCompletionSource 封装外部事件(如按钮点击、消息到达)
  • ManualResetEventSlim + WaitHandle.WaitOne(需注意线程上下文)
  • 在 async 方法中优先选 await Task.WhenAny(task, Task.Delay(timeoutMs)) 实现带超时的等待

记住:Delay(-1)WaitAll 都是“能跑通但不该用”的操作——它们暴露的是设计断点,而不是解决方案。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

313

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

522

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

49

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

190

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

173

2023.11.23

java中void的含义
java中void的含义

本专题整合了Java中void的相关内容,阅读专题下面的文章了解更多详细内容。

93

2025.11.27

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

473

2023.08.10

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

150

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

88

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号