推荐优先使用Task(尤其是async/await),因其更轻量、易管理、支持取消和异常传播;Thread更底层、开销大,仅适用于需精细控制线程生命周期的少数场景。

在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 写法简洁清晰。
常用方式:
千博购物系统.Net能够适合不同类型商品,为您提供了一个完整的在线开店解决方案。千博购物系统.Net除了拥有一般网上商店系统所具有的所有功能,还拥有着其它网店系统没有的许多超强功能。千博购物系统.Net适合中小企业和个人快速构建个性化的网上商店。强劲、安全、稳定、易用、免费是它的主要特性。系统由C#及Access/MS SQL开发,是B/S(浏览器/服务器)结构Asp.Net程序。多种独创的技术使
-
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("已取消"); }
Thread 和 Task 的关键区别
简单对比帮你做选择:
- Thread:每个实例对应一个操作系统线程,创建销毁成本高;无法返回值(需手动同步);无内置取消/等待机制
-
Task:逻辑任务,背后由线程池统一调度;支持泛型返回值(
Task);原生支持取消、延续(.ContinueWith)、组合(WhenAll) - 现代 C# 开发中,
Thread基本只用于特殊需求(如设置线程名称、设置 [STAThread]、或与非托管代码交互)
基本上就这些。日常开发用 Task.Run 和 async/await 足够覆盖绝大多数多线程和异步需求。









