C#怎么使用多线程 C# Thread与Task创建线程方法

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

在C#中,多线程主要通过 ThreadTask 两种方式实现,但推荐优先使用 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.Runasync 方法内部
  • 在任务中定期检查 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.Runasync/await 足够覆盖绝大多数多线程和异步需求。