推荐优先使用Task(尤其是async/await),因其更轻量、易管理、支持取消和异常传播;Thread更底层、开销大,仅适用于需精细控制线程生命周期的少数场景。
在C#中,多线程主要通过 Thread 和 Task 两种方式实现,但推荐优先使用 Task(尤其是 async/await),因为更轻量、易管理、支持取消和异常传播,而 Thread 更底层、开销大、已逐渐被替代。
Thread 是 .NET 最基础的线程封装,适合需要精细控制线程生命周期的极少数场景(如长时间运行的后台服务线程)。
基本用法:
Thread 实例,传入一个无参委托(ThreadStart)或带参委托(ParameterizedThreadStart)Start() 启动;可选传参(仅限一个 object)Abort()(已过时且不安全),应通过 CancellationToken 协作式退出示例:
var t = new Thread(() => {
Console.WriteLine("线程执行中...");
});
t.Start(); // 启动
t.Join(); // 等待结束(可选)Task 是基于线程池的异步抽象,无需手动管理线程,资源复用率高,配合 async/await 写法简洁清晰。
常用方式:
Task.Run(() => { ... }):将 CPU 密集型工作交给线程池执行Task.Factory.StartNew(...):更灵活(可配调度器、状态等),但一般用 Run 就够了async Task 方法 + await:处理 I/O 操作(如文件、网络、数据库)时真正不阻塞线程示例:
var task = Task.Run(() => {
Thread.Sleep(1000); // 模拟 CPU 工作
return 42;
});
int result = await task; // 或 task.Result(会阻塞,慎用)多线程场景下,取消和错误处理很关键。Task 天然支持 CancellationToken。
CancellationTokenSource,把 Token 传给 Task.Run 或 async 方法内部token.IsCancellationRequested,或调用 token.ThrowIfCancellationRequested()
Task.WhenAll(tasks) 等待多个任务完成;Task.WhenAny 获取最先完成的那个示例:
var cts = new CancellationTokenSource();
var task = Task.Run(() => {
for (int i =
0; i < 5; i++) {
Thread.Sleep(500);
cts.Token.ThrowIfCancellationRequested();
}
}, cts.Token);
// 3秒后取消
_ = Task.Run(() => { Thread.Sleep(3000); cts.Cancel(); });
try { await task; }
catch (OperationCanceledException) { Console.WriteLine("已取消"); }
简单对比帮你做选择:
Task);原生支持取消、延续(.ContinueWith)、组合(WhenAll)Thread 基本只用于特殊需求(如设置线程名称、设置 [STAThread]、或与非托管代码交互)基本上就这些。日常开发用 Task.Run 和 async/await 足够覆盖绝大多数多线程和异步需求。