在C#中实现多线程,推荐优先用 Task,而不是直接操作 Thread。Thread 更底层、难管理;Task 基于线程池,支持异步、取消、等待、异常传播和组合,是现代 C# 并发编程的标准方式。
Thread:简单但需手动管理
Thread 适合极少数需要完全控制线程生命周期的场景(如长时间运行的后台服务线程),但容易出错,不推荐日常使用。
创建后必须显式调用 Start() 才会执行 不能重复 Start;不能 Join 已结束的线程;无法返回值 异常不会自动向主线程传播,需在线程内 try-catch 处理 示例:var t = new Thread(() => {
Console.WriteLine("运行在线程:" + Thread.CurrentThread.ManagedThreadId);
});
t.Start();
t.Join(); // 等待完成
Task:默认首选,简洁可靠
Task 封装了线程调度逻辑,自动利用线程池,支持 await、ContinueWith、WaitAll/Any、取消令牌等关键能力。
启动即运行:Task.Run() 或 Task.Factory.StartNew() 会立即排队执行 可返回结果:用Task<t></t>,比如
Task<int> t = Task.Run(() => 42);</int>支持 async/await:真正实现非阻塞等待,避免线程浪费 统一异常处理:await 或 Wait() 时集中抛出所有异常(AggregateException)
推荐写法:用 Task.Run + async/await
这是最常用、最安全的模式,适用于 CPU 密集型工作(如计算、编码、解析)。
不要在 UI 线程(如 WinForms/WPF)中用 Thread.Sleep 或 long-running loop 用 await 避免阻塞界面,用 ConfigureAwait(false) 避免不必要的上下文捕获(类库中建议加) 示例:async Task<int> ComputeAsync()
{
return await Task.Run(() => {
Thread.Sleep(1000); // 模拟耗时计算
return 123;
});
}
<p>// 调用
int result = await ComputeAsync();
Console.WriteLine(result);
</p>进阶要点:取消与异常处理
真实项目中必须考虑任务可取消和错误恢复。
传入 CancellationToken 到 Task.Run 和内部逻辑中,主动响应取消请求 用 try/catch 包裹 await,捕获 OperationCanceledException(取消)或业务异常 多个任务并行?用 Task.WhenAll 或 Task.WhenAny,它们也支持 CancellationTokenvar cts = new CancellationTokenSource();
cts.CancelAfter(2000);
<p>try
{
var task = Task.Run(() => {
for (int i = 0; i < 10; i++)
{
cts.Token.ThrowIfCancellationRequested();
Thread.Sleep(500);
}
return "done";
}, cts.Token);</p><pre class="brush:php;toolbar:false;">string result = await task;} catch (OperationCanceledException) { Console.WriteLine("任务被取消"); }
基本上就这些。Thread 是“能用但不该用”,Task.Run + async/await 是“该用也够用”。别再手写 Thread.Start/Join/Abort,Task 已经帮你封装好所有复杂性。
