0

0

c# 在 C# 中,线程和纤程(Coroutine)的本质区别是什么

煙雲

煙雲

发布时间:2026-01-07 11:11:02

|

672人浏览过

|

来源于php中文网

原创

C#中无原生纤程,协程由async/await和IEnumerator模拟,运行于线程之上,协作式让出且无内核调度;线程由OS管理,可并行但开销大、需同步防护。

c# 在 c# 中,线程和纤程(coroutine)的本质区别是什么

线程是操作系统内核调度的执行单元,纤程(Coroutine)在 C# 中不是原生概念,而是由语言和运行时(如 async/awaitIEnumerator)模拟的用户态协作式执行流——它不对应 OS 线程,也不被内核调度。

线程由操作系统创建和管理,有独立和上下文

调用 new Thread(() => { ... }).Start() 会触发系统调用(如 Windows 的 CreateThread),分配约 1MB 栈空间(默认),并交由内核调度器决定何时运行、何时抢占。线程可并行(多核)或并发(单核时间片切换),但开销大、数量受限(几百个就可能耗尽内存或引发调度压力)。

常见错误现象:Thread.Abort() 已废弃,强行终止线程会导致资源泄漏或状态不一致;共享变量未加锁(lock / Interlocked)极易引发竞态。

  • 每个线程有独立的 ThreadLocal 存储
  • 线程异常未捕获会直接终止整个线程,不传播到启动方
  • 调试时可在 Visual Studio 的“线程”窗口中看到所有托管线程及其调用栈

C# 中没有“纤程”类型,只有协程模式的实现机制

所谓“纤程”在 C# 里实际指两类东西:IEnumerator(用于 yield return)和 Task(用于 async/await)。它们都运行在当前线程(通常是主线程或线程池线程)上,通过状态机(编译器生成的 g__ 类型)保存/恢复局部变量和执行位置,属于协作式让出(cooperative yielding),不涉及上下文切换开销。

容易混淆的点:Unity 的 StartCoroutine() 返回的是 Coroutine 对象,但它底层仍是基于 IEnumerator + 主循环驱动(每帧调用 MoveNext()),并非独立执行流。

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

下载
  • yield return null 让出控制权给同一线程的其他逻辑(如 UI 更新),不阻塞线程
  • await Task.Delay(100) 不阻塞线程,而是注册回调,由 SynchronizationContextThreadPool 在时机成熟时继续执行
  • 协程函数返回 IEnumeratorTask,本身不启动执行,需外部驱动(如 while (enumerator.MoveNext())await

调度模型与错误处理完全不同

线程崩溃(未捕获异常)会终止该线程,但不影响其他线程;而协程中的异常会立即抛出到驱动它的上下文——比如 async void 方法中未捕获异常会直接导致进程崩溃(AppDomain.UnhandledException),这是最常踩的坑。

async void BadHandler()
{
    await Task.Delay(100);
    throw new InvalidOperationException("Boom"); // 进程级崩溃!
}

// 正确写法:async Task,由调用方 await 并处理异常 async Task GoodHandler() { await Task.Delay(100); throw new InvalidOperationException("Boom"); // 可被 try/catch 捕获 }

  • 线程间通信靠 Queue + MonitorChannel;协程间靠 TaskCompletionSource 或事件回调
  • 线程可设置优先级(Thread.Priority),但协程完全无此概念——它的“时机”取决于状态机何时被调度执行
  • 使用 ConfigureAwait(false) 可避免协程回调强制回到原始上下文(如 UI 线程),提升性能并防止死锁

不要试图在 C# 中“手动实现纤程调度器”

有人尝试用 Thread.Yield() + 循环 + 状态标记模拟纤程,这既无必要又危险:C# 的 async/await 编译器已为你生成高效状态机,且与 ThreadPoolSynchronizationContext 深度集成。手写调度器大概率破坏异步流的取消传播(CancellationToken)、超时控制和诊断能力(如 dotnet trace 无法识别自定义状态流转)。

真正需要区分的不是“线程 vs 纤程”,而是“是否需要并行执行”和“是否需要等待 I/O 而不阻塞线程”。前者选线程(或更现代的 Parallel / PLINQ),后者一律用 async/await

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

435

2024.03.01

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

83

2023.09.25

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

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

174

2023.11.23

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

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

95

2025.11.27

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

382

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

567

2023.08.10

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

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

476

2023.08.10

java学习网站推荐汇总
java学习网站推荐汇总

本专题整合了java学习网站相关内容,阅读专题下面的文章了解更多详细内容。

6

2026.01.08

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 3.5万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

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

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