c# 异步( Async ) 不是多线程
误解 async
在调试 xxxxAsync() 方法的时候,常常会看到调试器界面中会多出一些线程,直觉上误认为 Async 冠名的函数是多线程。 对于 StringReader 中的 ReadAsync() 方法的确如此,
ReadAsync() 在内部调用 Task.FromResult(),而 Task 正是微软提供封装完整的线程池。但是对于 HttpClient 中的 GetStringAsyncCore() 则是由 async 修饰的异步函数。
由于多线程和异步函数都可能冠上 Async 因此,一开始容易误以为 async 等于 异步。在多线程或异步函数上冠名 Async 则是一种默认的命名规范。异步、多线程的区别
异步 : 属于通信的范畴,在发出消息当下不等待对方回应,便开始继续自己的任务。所以 c# 的很多 _异步_ 操作都在 io 、 socket 这些类库下面,都是通信性质的类库,io 可以理解向系统内核发送文件数据
多线程 : 属于计算范畴,通常是关于如果利用 cpu 的空闲时间进行计算。
异步同多线程可以完美的结合,服务器开启多个线程监听前端请求,接收处理完毕后,可以异步地将数据序列化到磁盘,当然如果需要知道序列化的结果则异步处理可以等待 ( await )。
从这个例子不难知道,多线程是提高数据的计算能力,而异步则是为了提高程序的吞吐量。
异步的硬件支持
计算机硬件需要依赖中断来请求 cpu 进行计算,但是中断需要存储上下文而且耗费性能,不能频繁中断 cpu 。当进行 io 操作时,一般会在用户态上进行缓存,这时数据在内存上,接下来 cpu 切到系统态上,让 Direct Memory Access (DMA) 完成数据传输,这时 cpu 完全不过问 io 操作,待 DMA 完成时,会将在寄存器上做标记。
异步的程序原理
异步是将程序分成几个部分,每部分由一个状态控制,最后形成一个有限状态机。