同步(Synchronous)和异步(Asynchronous)

上一篇多线程

C#多线程-CSDN博客

同步特点

  • 代码按顺序执行,一个操作完成后才会执行下一个。
  • 阻塞当前线程,直到操作完成(例如:I/O、网络请求、计算等)。
  • 简单直观,但可能导致性能问题(例如:UI冻结、服务器吞吐量低等)。
cs 复制代码
using System.Net.Http;
using System.Threading.Tasks;

public string GetWebsiteContentSync(string url)
{
    using (HttpClient client = new HttpClient())
    {
        return client.GetStringAsync(url).Result; // 这里使用.Result会导致同步等待
    }
}

异步特点

  • 非阻塞执行,允许在等待操作完成时释放当前线程去做其他工作。
  • 提高响应性(如UI不冻结)和吞吐量(如服务器处理更多请求)。
  • 通过Async/await关键字实现,依赖Task或Taks<T>类型。
cs 复制代码
public async Task<string> GetWebsiteContentAsync(string url)
{
    using (HttpClient client = new HttpClient())
    {
        return await client.GetStringAsync(url); // 使用await进行异步等待
    }
}
async/await
  • `async`:标记方法为异步方法,方法内可使用`await`。
  • `await`:挂起当前方法,直到异步操作完成,期间线程可被释放去做其他任务。
  • 异步方法应返回`Task`、`Task<T>`或`ValueTask(高性能场景)`。
Task 和 Task<T>
  • Task:表示一个异步操作,可能无返回值。
  • Task<T>:表示返回类型为 `T` 的异步操作。
  • 使用 Task.Run 可将同步代码包装为异步(慎用,避免滥用线程池)。

异步编程最佳实践

避免阻塞异步代码
  • ❌ 不要使用 .Result 或 .Wait(),可能导致死锁。
  • ✅ 始终使用 await:
cs 复制代码
// 正确方式
var result = await SomeAsyncMethod();
异常处理

异步方法中的异常会被包装到Task中,可通过try/catch捕捉:

cs 复制代码
try
{
    await SomeAsyncMethod();
}
catch (Exception ex)
{
    // 处理异常
}
配置上下文

在库代码中使用ConfigureAwait(false)避免上下文切换:

cs 复制代码
var data = await SomeAsyncMethod().ConfigureAwait(false);
避免 async void

仅限事件处理程序(如:按钮点击),否则异常会导致进程崩溃:

cs 复制代码
private async void Button_Click(object sender, EventArgs e)
{
    await SomeAsyncMethod();
}

同步和异步使用场景

场景 同步 异步
UI 响应性(如 WinForms、WPF) ❌ 导致界面冻结 ✅ 保持界面流畅
服务端处理(如 ASP.NET ❌ 吞吐量低 ✅ 高并发处理
I/O 密集型操作(文件、网络) ❌ 阻塞线程 ✅ 释放线程
CPU 密集型计算 ✅ 直接执行 ❌ 需配合 Task.Run
常见误区
  1. 虚假异步:方法标记为 `async` 但内部无 `await`(编译器警告)。

  2. 死锁:在同步上下文中调用 `.Result` 或 `.Wait()`。

  3. 过度异步化:滥用 `Task.Run` 包装同步代码,增加线程切换开销。

总结

  • 同步:简单但阻塞,适合快速完成的操作。
  • 异步:非阻塞且高效,适合 I/O 或高延迟操作。
  • 始终遵循 async/await 模式,避免阻塞调用,合理处理异常和上下文。
相关推荐
Morwit2 分钟前
Qt qml创建c++类的单例对象
开发语言·c++·qt
古城小栈4 分钟前
Rust 已经自举,却仍需GNU与MSVC工具链的缘由
开发语言·rust
jarreyer9 分钟前
数据项目分析标准化流程
开发语言·python·机器学习
你怎么知道我是队长12 分钟前
C语言---printf函数使用详细说明
c语言·开发语言
liulilittle14 分钟前
俄罗斯访问欧洲国际线路优化
开发语言·网络·信息与通信·ip·通信·俄罗斯·莫斯科
陈小桔16 分钟前
logging模块-python
开发语言·python
消失的旧时光-194317 分钟前
函数指针 + 结构体 = C 语言的“对象模型”?——从 C 到 C++ / Java 的本质统一
linux·c语言·开发语言·c++·c
!停18 分钟前
C语言栈和队列的实现
开发语言·数据结构
hixiong12318 分钟前
C# OpenvinoSharp部署DDDDOCR验证码识别模型
opencv·c#·ocr·openvino
源代码•宸19 分钟前
Golang语法进阶(定时器)
开发语言·经验分享·后端·算法·golang·timer·ticker