C#多线程与异步的区别详解

发布时间 - 2026-01-11 01:45:45    点击率:

C#多线程与异步的区别详解

随着拥有多个硬线程 CPU(超线程、双核)的普及,多线程和异步操作等并发程序设计方法也受到了更多的关注和讨论。本文主要是想与各位高手一同探讨一下如何使用并发来最大化程序的性能。

多线程和异步操作的异同

多线程和异步操作两者都可以达到避免调用线程阻塞的目的,从而提高软件的可响应性。甚至有些时候我们就认为多线程和异步操作是等同的概念。但是,多线程和异步操作还是有一些区别的。而这些区别造成了使用多线程和异步操作的时机的区别。

异步操作的本质

所有的程序最终都会由计算机硬件来执行,所以为了更好的理解异步操作的本质,我们有必要了解一下它的硬件基础。 熟悉电脑硬件的朋友肯定对 DMA 这个词不陌生,硬盘、光驱的技术规格中都有明确 DMA 的模式指标,其实网卡、声卡、显卡也是有 DMA 功能的。DMA 就是直接内存访问的意思,也就是说,拥有 DMA 功能的硬件在和内存进行数据交换的时候可以不消耗 CPU 资源。只要 CPU 在发起数据传输时发送一个指令,硬件就开始自己和内存交换数据,在传输完成之后硬件会触发一个中断来通知操作完成。这些无须消耗 CPU 时间的 I/O 操作正是异步操作的硬件基础。所以即使在 DOS 这样的单进程(而且无线程概念)系统中也同样可以发起异步的 DMA 操作。

线程的本质

线程不是一个计算机硬件的功能,而是操作系统提供的一种逻辑功能,线程本质上是进程中一段并发运行的代码,所以线程需要操作系统投入 CPU 资源来运行和调度。

异步操作的优点与缺点

因为异步操作无须额外的线程负担,并且使用回调的方式进行处理,在设计良好的情况下,处理函数可以不必使用共享变量(即使无法完全不用,最起码可以减少共享变量的数量),减少了死锁的可能。当然异步操作也并非完美无暇。编写异步操作的复杂程度较高,程序主要使用回调方式进行处理,与普通人的思维方式有些初入,而且难以调试。

多线程的优点与缺点

多线程的优点很明显,线程中的处理程序依然是顺序执行,符合普通人的思维习惯,所以编程简单。但是多线程的缺点也同样明显,线程的使用(滥用)会给系统带来上下文切换的额外负担。并且线程间的共享变量可能造成死锁的出现。

适用范围

在了解了线程与异步操作各自的优点与缺点之后,我们可以来探讨一下线程和异步的合理用途。我认为:当需要执行 I/O 操作时,使用异步操作比使用线程加同步 I/O 操作更合适。I/O 操作不仅包括了直接的文件、网络的读写,还包括数据库操作、Web Service、HttpRequest 以及 .Net Remoting 等跨进程的调用。

而线程的适用范围则是那种需要长时间 CPU 运算的场合,例如耗时较长的图形处理和算法执行。但是往往由于使用线程编程的简单和符合习惯,所以很多朋友往往会使用线程来执行耗时较长的 I/O 操作。这样在只有少数几个并发操作的时候还无伤大雅,如果需要处理大量的并发操作时就不合适了。

实例研究

说了那么理论上的东西,可能有些兄弟早就不耐烦了,现在我们来研究几个实际的异步操作例子吧。

实例1:由 delegate 产生的异步方法到底是怎么回事?

大家可能都知道,使用 delegate 可以“自动”使一个方法可以进行异步的调用。从直觉上来说,我觉得是由编译器或者 CLR 使用了另外的线程来执行目标方法。到底是不是这样呢?让我们来用一段代码证明一下吧。

using System;
using System.Threading;
 
namespace AsyncDelegateDemo
{
 delegate void AsyncFoo(int i);
 class Program
 {
  ///<summary>
  /// 输出当前线程的信息
  ///</summary>
  ///<param name="name">方法名称</param>
 
  static void PrintCurrThreadInfo(string name)
  {
   Console.WriteLine("Thread Id of " + name + " is: " + Thread.CurrentThread.ManagedThreadId + ", current thread is "
            + (Thread.CurrentThread.IsThreadPoolThread ? "" : "not ")
            + "thread pool thread.");
  }
 
  ///<summary>
  /// 测试方法,Sleep一定时间
  ///</summary>
  ///<param name="i">Sleep的时间</param>
  static void Foo(int i)
  {
   PrintCurrThreadInfo("Foo()");
   Thread.Sleep(i);
  }
 
  ///<summary>
  /// 投递一个异步调用
  ///</summary>
  static void PostAsync()
  {
   AsyncFoo caller = new AsyncFoo(Foo);
   caller.BeginInvoke(1000, new AsyncCallback(FooCallBack), caller);
  }
 
  static void Main(string[] args)
  {
   PrintCurrThreadInfo("Main()");
   for(int i = 0; i < 10 ; i++)
   {
    PostAsync();
   }
   Console.ReadLine();
  }
 
  static void FooCallBack(IAsyncResult ar)
  {
   PrintCurrThreadInfo("FooCallBack()");
   AsyncFoo caller = (AsyncFoo) ar.AsyncState;
   caller.EndInvoke(ar);
  }
 }
}

这段代码代码的输出如下:

Thread Id of Main() is: 1, current thread is not thread pool thread.
Thread Id of Foo() is: 3, current thread is thread pool thread.
Thread Id of FooCallBack() is: 3, current thread is thread pool thread.
Thread Id of Foo() is: 3, current thread is thread pool thread.
Thread Id of Foo() is: 4, current thread is thread pool thread.
Thread Id of Foo() is: 5, current thread is thread pool thread.
Thread Id of FooCallBack() is: 3, current thread is thread pool thread.
Thread Id of Foo() is: 3, current thread is thread pool thread.
Thread Id of FooCallBack() is: 4, current thread is thread pool thread.
Thread Id of Foo() is: 4, current thread is thread pool thread.
Thread Id of Foo() is: 6, current thread is thread pool thread.
Thread Id of FooCallBack() is: 5, current thread is thread pool thread.
Thread Id of Foo() is: 5, current thread is thread pool thread.
Thread Id of Foo() is: 7, current thread is thread pool thread.
Thread Id of FooCallBack() is: 3, current thread is thread pool thread.
Thread Id of Foo() is: 3, current thread is thread pool thread.
Thread Id of FooCallBack() is: 4, current thread is thread pool thread.
Thread Id of FooCallBack() is: 6, current thread is thread pool thread.
Thread Id of FooCallBack() is: 5, current thread is thread pool thread.
Thread Id of FooCallBack() is: 7, current thread is thread pool thread.
Thread Id of FooCallBack() is: 3, current thread is thread pool thread.

从输出可以看出,.net 使用 delegate 来“自动”生成的异步调用是使用了另外的线程(而且是线程池线程)。

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


# C#多线程与异步的区别  # C#  # 多线程与异步对比  # C# 异步多线程入门到精通之ThreadPool篇  # C# 异步多线程入门到精通之Thread篇  # C# 异步多线程入门基础  # 深入分析C#中的异步和多线程  # 解析C#多线程编程中异步多线程的实现及线程池的使用  # 详解C#异步多线程使用中的常见问题  # 多线程  # 死锁  # 几个  # 较长  # 回调  # 计算机硬件  # 操作系统  # 使用了  # 都有  # 我觉得  # 是有  # 让我们  # 多个  # 说了  # 无伤大雅  # 是由  # 则是  # 长时间  # 较高  # 这段 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何实现全文搜索_Laravel Scout集成Algolia或Meilisearch教程  如何在万网主机上快速搭建网站?  Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例  Swift开发中switch语句值绑定模式  php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】  Laravel如何安装Breeze扩展包_Laravel用户注册登录功能快速实现【流程】  Laravel怎么进行数据库回滚_Laravel Migration数据库版本控制与回滚操作  Laravel如何实现密码重置功能_Laravel密码找回与重置流程  如何快速搭建虚拟主机网站?新手必看指南  如何在橙子建站上传落地页?操作指南详解  HTML5段落标签p和br怎么选_文本排版常用标签对比【解答】  北京网页设计制作网站有哪些,继续教育自动播放怎么设置?  制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?  Laravel中间件如何使用_Laravel自定义中间件实现权限控制  Python结构化数据采集_字段抽取解析【教程】  linux写shell需要注意的问题(必看)  Laravel怎么实现搜索高亮功能_Laravel结合Scout与Algolia全文检索【实战】  Laravel怎么进行浏览器测试_Laravel Dusk自动化浏览器测试入门  微信小程序 HTTPS报错整理常见问题及解决方案  javascript如何操作浏览器历史记录_怎样实现无刷新导航  如何在橙子建站中快速调整背景颜色?  如何快速启动建站代理加盟业务?  品牌网站制作公司有哪些,买正品品牌一般去哪个网站买?  Laravel如何使用Facades(门面)及其工作原理_Laravel门面模式与底层机制  教你用AI将一段旋律扩展成一首完整的曲子  Python文件操作最佳实践_稳定性说明【指导】  轻松掌握MySQL函数中的last_insert_id()  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  rsync同步时出现rsync: failed to set times on “xxxx”: Operation not permitted  Laravel怎么判断请求类型_Laravel Request isMethod用法  绝密ChatGPT指令:手把手教你生成HR无法拒绝的求职信  如何自定义safari浏览器工具栏?个性化设置safari浏览器界面教程【技巧】  WEB开发之注册页面验证码倒计时代码的实现  高防服务器租用如何选择配置与防御等级?  Laravel广播系统如何实现实时通信_Laravel Reverb与WebSockets实战教程  html5怎么画眼睛_HT5用Canvas或SVG画眼球瞳孔加JS控制动态【绘制】  Laravel Seeder填充数据教程_Laravel模型工厂Factory使用  Laravel怎么实现支付功能_Laravel集成支付宝微信支付  Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】  百度浏览器网页无法复制文字怎么办 百度浏览器复制修复  微信小程序 wx.uploadFile无法上传解决办法  如何快速查询域名建站关键信息?  详解Android中Activity的四大启动模式实验简述  如何在IIS中配置站点IP、端口及主机头?  jQuery validate插件功能与用法详解  如何在IIS7中新建站点?详细步骤解析  Laravel如何实现事件和监听器?(Event & Listener实战)  HTML5打空格有哪些误区_新手常犯的空格使用错误【技巧】  如何用好域名打造高点击率的自主建站?  中山网站推广排名,中山信息港登录入口?