在C#中,多线程主要通过
Thread和
Task两种方式实现,但推荐优先使用
Task(尤其是
async/await),因为更轻量、易管理、支持取消和异常传播,而
Thread更底层、开销大、已逐渐被替代。
直接用 Thread 创建和启动线程
Thread是 .NET 最基础的线程封装,适合需要精细控制线程生命周期的极少数场景(如长时间运行的后台服务线程)。
基本用法:
创建Thread实例,传入一个无参委托(
ThreadStart)或带参委托(
ParameterizedThreadStart) 调用
Start()启动;可选传参(仅限一个 object) 不建议手动调用
Abort()(已过时且不安全),应通过
CancellationToken协作式退出
示例:
var t = new Thread(() => {
Console.WriteLine("线程执行中...");
});
t.Start(); // 启动
t.Join(); // 等待结束(可选)用 Task 启动并发操作(推荐)
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);
<p>// 3秒后取消
_ = Task.Run(() => { Thread.Sleep(3000); cts.Cancel(); });</p><p>try { await task; }
catch (OperationCanceledException) { Console.WriteLine("已取消"); }Thread 和 Task 的关键区别
简单对比帮你做选择:
Thread:每个实例对应一个操作系统线程,创建销毁成本高;无法返回值(需手动同步);无内置取消/等待机制 Task:逻辑任务,背后由线程池统一调度;支持泛型返回值(Task<t></t>);原生支持取消、延续(
.ContinueWith)、组合(
WhenAll) 现代 C# 开发中,
Thread基本只用于特殊需求(如设置线程名称、设置 [STAThread]、或与非托管代码交互)
基本上就这些。日常开发用
Task.Run和
async/await足够覆盖绝大多数多线程和异步需求。
