C#多线程和异步(二)——Task和async/await详解

一、什么是异步

  同步和异步主要用于修饰方法。当一个方法被调用时,调用者需要等待该方法执行完毕并返回才能继续执行,我们称这个方法是同步方法;当一个方法被调用时立即返回,并获取一个线程执行该方法内部的业务,调用者不用等待该方法执行完毕,我们称这个方法为异步方法。

  异步的好处在于非阻塞(调用线程不会暂停执行去等待子线程完成),因此我们把一些不需要立即使用结果、较耗时的任务设为异步执行,可以提高程序的运行效率。net4.0在ThreadPool的基础上推出了Task类,微软极力推荐使用Task来执行异步任务,现在C#类库中的异步方法基本都用到了Task;net5.0推出了async/await,让异步编程更为方便。本篇主要介绍Task、async/await相关的内容,其他异步操作的方式会在下一篇介绍。

回到顶部

二、Task介绍

  Task是在ThreadPool的基础上推出的,我们简单了解下ThreadPool。ThreadPool中有若干数量的线程,如果有任务需要处理时,会从线程池中获取一个空闲的线程来执行任务,任务执行完毕后线程不会销毁,而是被线程池回收以供后续任务使用。当线程池中所有的线程都在忙碌时,又有新任务要处理时,线程池才会新建一个线程来处理该任务,如果线程数量达到设置的最大值,任务会排队,等待其他任务释放线程后再执行。线程池能减少线程的创建,节省开销,看一个ThreadPool的栗子吧

static void Main(string[] args){for (int i = 1; i <=10; i++){//ThreadPool执行任务ThreadPool.QueueUserWorkItem(new WaitCallback((obj) => {Console.WriteLine($"第{obj}个执行任务");}),i);}Console.ReadKey();}

  上边的代码通过ThreadPool执行了10个任务,执行结果为:

   ThreadPool相对于Thread来说可以减少线程的创建,有效减小系统开销;但是ThreadPool不能控制线程的执行顺序,我们也不能获取线程池内线程取消/异常/完成的通知,即我们不能有效监控和控制线程池中的线程。

1 Task创建和运行

  我们知道了ThreadPool的弊端:我们不能控制线程池中线程的执行顺序,也不能获取线程池内线程取消/异常/完成的通知。net4.0在ThreadPool的基础上推出了Task,Task拥有线程池的优点,同时也解决了使用线程池不易控制的弊端。

首先看一下怎么去创建并运行一个Task,Task的创建和执行方式有如下三种:

static void Main(string[] args){//1.new方式实例化一个Task,需要通过Start方法启动Task task = new Task(() =>{Thread.Sleep(100);Console.WriteLine($"hello, task1的线程ID为{Thread.CurrentThread.ManagedThreadId}");});task.Start();//2.Task.Factory.StartNew(Action action)创建和启动一个TaskTask task2 = Task.Factory.StartNew(() =>{Thread.Sleep(100);Console.WriteLine($"hello, task2的线程ID为{ Thread.CurrentThread.ManagedThreadId}");});//3.Task.Run(Action action)将任务放在线程池队列,返回并启动一个TaskTask task3 = Task.Run(() =>{Thread.Sleep(100);Console.WriteLine($"hello, task3的线程ID为{ Thread.CurrentThread.ManagedThreadId}");});Console.WriteLine("执行主线程!");Console.ReadKey();}

执行结果如下:

  我们看到先打印"执行主线程",然后再打印各个任务,说明了Task不会阻塞主线程。上边的栗子Task都没有返回值,我们也可以创建有返回值的Task<TResult>,用法和没有返回值的基本一致,我们简单修改一下上边的栗子,代码如下:

static void Main(string[] args){1.new方式实例化一个Task,需要通过Start方法启动Task<string> task = new Task<string>(() =>{return $"hello, task1的ID为{Thread.CurrentThread.ManagedThreadId}";});task.Start();2.Task.Factory.StartNew(Func func)创建和启动一个TaskTask<string> task2 =Task.Factory.StartNew<string>(() =>{return $"hello, task2的ID为{ Thread.CurrentThread.ManagedThreadId}";});3.Task.Run(Func func)将任务放在线程池队列,返回并启动一个TaskTask<string> task3= Task.Run<string>(() =>{return $"hello, task3的ID为{ Thread.CurrentThread.ManagedThreadId}";});Console.WriteLine("执行主线程!");Console.WriteLine(task.Result);Console.WriteLine(task2.Result);Console.WriteLine(task3.Result);Console.ReadKey();}

  注意task.Resut获取结果时会阻塞线程,即如果task没有执行完成,会等待task执行完成获取到Result,然后再执行后边的代码,程序运行结果如下:   

  上边的所有栗子中Task的执行都是异步的,不会阻塞主线程。有些场景下我们想让Task同步执行怎么办呢?Task提供了  task.RunSynchronously()用于同步执行Task任务,代码如下:

static void Main(string[] args){Task task = new Task(() =>{Thread.Sleep(100);Console.WriteLine("执行Task结束!");});//同步执行,task会阻塞主线程task.RunSynchronously();Console.WriteLine("执行主线程结束!");Console.ReadKey();}

执行结果如下:

 2 Task的阻塞方法(Wait/WaitAll/WaitAny)

1 Thread阻塞线程的方法

  使用Thread时,我们知道用thread.Join()方法即可阻塞主线程。看一个例子:

static void Main(string[] args){Thread th1 = new Thread(() => {Thread.Sleep(500);Console.WriteLine("线程1执行完毕!");});th1.Start();Thread th2 = new Thread(() => {Thread.Sleep(1000);Console.WriteLine("线程2执行完毕!");});th2.Start();//阻塞主线程th1.Join();th2.Join();Console.WriteLine("主线程执行完毕!");Console.ReadKey();}

  如果注释掉两个Join,执行结果是:先打印【主线程执行完毕】,而添加两个Join方法后执行结果如下,实现了线程阻塞:

2 Task的Wait/WaitAny/WaitAll方法

   Thread的Join方法可以阻塞调用线程,但是有一些弊端:①如果我们要实现很多线程的阻塞时,每个线程都要调用一次Join方法;②如果我们想让所有的线程执行完毕(或者任一线程执行完毕)时,立即解除阻塞,使用Join方法不容易实现。Task提供了  Wait/WaitAny/WaitAll  方法,可以更方便地控制线程阻塞。

  task.Wait()  表示等待task执行完毕,功能类似于thead.Join();  Task.WaitAll(Task[] tasks)  表示只有所有的task都执行完成了再解除阻塞;  Task.WaitAny(Task[] tasks) 表示只要有一个task执行完毕就解除阻塞,看一个栗子: 

        static void Main(string[] args){Task task1 = new Task(() => {Thread.Sleep(500);Console.WriteLine("线程1执行完毕!");});task1.Start();Task task2 = new Task(() => {Thread.Sleep(1000);Console.WriteLine("线程2执行完毕!");});task2.Start();//阻塞主线程。task1,task2都执行完毕再执行主线程//执行【task1.Wait();task2.Wait();】可以实现相同功能Task.WaitAll(new Task[]{ task1,task2});Console.WriteLine("主线程执行完毕!");Console.ReadKey();}

  执行结果如下:

  如果将栗子中的WaitAll换成WaitAny,那么任一task执行完毕就会解除线程阻塞,执行结果是:先打印【线程1执行完毕】,然后打印【主线程执行完毕】,最后打印【线程2执行完毕】

 3 Task的延续操作(WhenAny/WhenAll/ContinueWith)

  上边的Wait/WaitAny/WaitAll方法返回值为void,这些方法单纯的实现阻塞线程。我们现在想让所有task执行完毕(或者任一task执行完毕)后,开始执行后续操作,怎么实现呢?这时就可以用到WhenAny/WhenAll方法了,这些方法执行完成返回一个task实例。  task.WhenAll(Task[] tasks)  表示所有的task都执行完毕后再去执行后续的操作, task.WhenAny(Task[] tasks)  表示任一task执行完毕后就开始执行后续操作。看一个栗子:

static void Main(string[] args){Task task1 = new Task(() => {Thread.Sleep(500);Console.WriteLine("线程1执行完毕!");});task1.Start();Task task2 = new Task(() => {Thread.Sleep(1000);Console.WriteLine("线程2执行完毕!");});task2.Start();//task1,task2执行完了后执行后续操作Task.WhenAll(task1, task2).ContinueWith((t) => {Thread.Sleep(100);Console.WriteLine("执行后续操作完毕!");});Console.WriteLine("主线程执行完毕!");Console.ReadKey();}

  执行结果如下,我们看到WhenAll/WhenAny方法不会阻塞主线程,当使用WhenAll方法时所有的task都执行完毕才会执行后续操作;如果把栗子中的WhenAll替换成WhenAny,则只要有一个线程执行完毕就会开始执行后续操作,这里不再演示。

   上边的栗子也可以通过 Task.Factory.ContinueWhenAll(Task[] tasks, Action continuationAction)  和 Task.Factory.ContinueWhenAny(Task[] tasks, Action continuationAction) 来实现 ,修改上边栗子代码如下,执行结果不变。

static void Main(string[] args){Task task1 = new Task(() => {Thread.Sleep(500);Console.WriteLine("线程1执行完毕!");});task1.Start();Task task2 = new Task(() => {Thread.Sleep(1000);Console.WriteLine("线程2执行完毕!");});task2.Start();//通过TaskFactroy实现Task.Factory.ContinueWhenAll(new Task[] { task1, task2 }, (t) =>{Thread.Sleep(100);Console.WriteLine("执行后续操作");});Console.WriteLine("主线程执行完毕!");Console.ReadKey();}

 4 Task的任务取消(CancellationTokenSource)

1 Thread取消任务执行

  在Task前我们执行任务采用的是Thread,Thread怎么取消任务呢?一般流程是:设置一个变量来控制任务是否停止,如设置一个变量isStop,然后线程轮询查看isStop,如果isStop为true就停止,代码如下:

static void Main(string[] args){bool isStop = false;int index = 0;//开启一个线程执行任务Thread th1 = new Thread(() =>{while (!isStop){Thread.Sleep(1000);Console.WriteLine($"第{++index}次执行,线程运行中...");}});th1.Start();//五秒后取消任务执行Thread.Sleep(5000);isStop = true;Console.ReadKey();}

2 Task取消任务执行

  Task中有一个专门的类 CancellationTokenSource  来取消任务执行,还是使用上边的例子,我们修改代码如下,程序运行的效果不变。

        static void Main(string[] args){CancellationTokenSource source = new CancellationTokenSource();int index = 0;//开启一个task执行任务Task task1 = new Task(() =>{while (!source.IsCancellationRequested){Thread.Sleep(1000);Console.WriteLine($"第{++index}次执行,线程运行中...");}});task1.Start();//五秒后取消任务执行Thread.Sleep(5000);//source.Cancel()方法请求取消任务,IsCancellationRequested会变成truesource.Cancel();Console.ReadKey();}

   CancellationTokenSource的功能不仅仅是取消任务执行,我们可以使用  source.CancelAfter(5000)  实现5秒后自动取消任务,也可以通过  source.Token.Register(Action action)  注册取消任务触发的回调函数,即任务被取消时注册的action会被执行。看一个栗子:

static void Main(string[] args){CancellationTokenSource source = new CancellationTokenSource();//注册任务取消的事件source.Token.Register(() =>{Console.WriteLine("任务被取消后执行xx操作!");});int index = 0;//开启一个task执行任务Task task1 = new Task(() =>{while (!source.IsCancellationRequested){Thread.Sleep(1000);Console.WriteLine($"第{++index}次执行,线程运行中...");}});task1.Start();//延时取消,效果等同于Thread.Sleep(5000);source.Cancel();source.CancelAfter(5000);Console.ReadKey();}

  执行结果如下,第5次执行在取消回调后打印,这是因为,执行取消的时候第5次任务已经通过了while()判断,任务已经执行中了:

   最后看上一篇跨线程的栗子,点击按钮启动一个任务,给tetxtbox赋值,我们把Thread改成Task,代码如下:

public partial class Form1 : Form{public Form1(){InitializeComponent();}private void mySetValueBtn_Click(object sender, EventArgs e){Task.Run(() =>{Action<int> setValue = (i) => { myTxtbox.Text = i.ToString(); };for (int i = 0; i < 1000000; i++){myTxtbox.Invoke(setValue,i);}});}}

  运行界面如下,赋值的task不会阻塞UI线程:

回到顶部

三、异步方法(async/await)

  在C#5.0中出现的 async和await ,让异步编程变得更简单。我们看一个获取文件内容的栗子:

class Program{static void Main(string[] args){string content = GetContentAsync(Environment.CurrentDirectory + @"/test.txt").Result;//调用同步方法//string content = GetContent(Environment.CurrentDirectory + @"/test.txt");Console.WriteLine(content);Console.ReadKey();}//异步读取文件内容async static Task<string> GetContentAsync(string filename){FileStream fs = new FileStream(filename, FileMode.Open);var bytes = new byte[fs.Length];//ReadAync方法异步读取内容,不阻塞线程Console.WriteLine("开始读取文件");int len = await fs.ReadAsync(bytes, 0, bytes.Length);string result = Encoding.UTF8.GetString(bytes);return result;}//同步读取文件内容static string GetContent(string filename){FileStream fs = new FileStream(filename, FileMode.Open);var bytes = new byte[fs.Length];//Read方法同步读取内容,阻塞线程int len =  fs.Read(bytes, 0, bytes.Length);string result = Encoding.UTF8.GetString(bytes);return result;}}

  test.txt内容是【hello world!】执行结果为:

  上边的栗子也写出了同步读取的方式,将main函数中的注释去掉即可同步读取文件内容。我们可以看到异步读取代码和同步读取代码基本一致。async/await让异步编码变得更简单,我们可以像写同步代码一样去写异步代码。注意一个小问题:异步方法中方法签名返回值为Task<T>,代码中的返回值为T。上边栗子中GetContentAsync的签名返回值为Task<string>,而代码中返回值为string。牢记这一细节对我们分析异步代码很有帮助。

  异步方法签名的返回值有以下三种:

    ① Task<T>:如果调用方法想通过调用异步方法获取一个T类型的返回值,那么签名必须为Task<TResult>;

    ② Task:如果调用方法不想通过异步方法获取一个值,仅仅想追踪异步方法的执行状态,那么我们可以设置异步方法签名的返回值为Task;

    ③ void:如果调用方法仅仅只是调用一下异步方法,不和异步方法做其他交互,我们可以设置异步方法签名的返回值为void,这种形式也叫做“调用并忘记”。

  小结:到这里Task,async/await的简单使用已经基本结束了,一些高级特性等到工作遇到了再去研究。通过上边的介绍,我们知道async/await是基于Task的,而Task是对ThreadPool的封装改进,主要是为了更有效的控制线程池中的线程(ThreadPool中的线程,我们很难通过代码控制其执行顺序,任务延续和取消等等);ThreadPool基于Thread的,主要目的是减少Thread创建数量和管理Thread的成本。async/await Task是C#中更先进的,也是微软大力推广的特性,我们在开发中可以尝试使用Task来替代Thread/ThreadPool,处理本地IO和网络IO任务是尽量使用async/await来提高任务执行效率。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/304951.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

从头到尾彻底理解傅里叶变换算法(上)

从头到尾彻底理解傅里叶变换算法&#xff08;上&#xff09; 前言 第一部分、 DFT 第一章、傅立叶变换的由来 第二章、实数形式离散傅立叶变换&#xff08;Real DFT&#xff09; 从头到尾彻底理解傅里叶变换算法、下 第三章、复数 第四章、复数形式离散傅立叶变换 前言&#x…

使用ADO.NET的参数集合来有效防止SQL注入漏洞

SQL注入漏洞是个老话题了&#xff0c;在以前做ASP做开发时&#xff0c;就经常需要用字符串的过虑等方式来解决这个问题&#xff0c;但有时候确做的不够彻底&#xff0c;往往让***钻了空子。那么目前在我们.NET中&#xff0c;不管是用WINFORM开发还是用WEBFORM&#xff0c;连接数…

[Abp 源码分析]ASP.NET Core 集成

点击上方蓝字关注我们0. 简介整个 Abp 框架最为核心的除了 Abp 库之外&#xff0c;其次就是 Abp.AspNetCore 库了。虽然 Abp 本身是可以用于控制台程序的&#xff0c;不过那样的话 Abp 就基本没什么用&#xff0c;还是需要集合 ASP.NET Core 才能发挥它真正的作用。在 Abp.AspN…

从头到尾彻底理解傅里叶变换算法(下)

从头到尾彻底理解傅里叶变换算法&#xff08;上&#xff09;&#xff0c;请看今天第一条。 以下继续&#xff1a; 第三章、复数 复数扩展了我们一般所能理解的数的概念&#xff0c;复数包含了实数和虚数两部分&#xff0c;利用复数的形式可以把由两个变量表示的表达式变成由一个…

树莓派安装python3.5_梦见树_周公解梦梦到树是什么意思_做梦梦见树好不好_周公解梦官网...

梦见树是什么意思&#xff1f;做梦梦见树好不好&#xff1f;梦见树有现实的影响和反应&#xff0c;也有梦者的主观想象&#xff0c;请看下面由(周公解梦官网www.zgjm.org)小编帮你整理的梦见树的详细解说吧。树主健康&#xff0c;树笔直挺拔&#xff0c;象征着人的健康。 在梦中…

[Abp 源码分析]后台作业与后台工作者

点击上方蓝字关注我们0. 简介在某些时候我们可能会需要执行后台任务&#xff0c;或者是执行一些周期性的任务。比如说可能每隔 1 个小时要清除某个临时文件夹内的数据&#xff0c;可能用户会要针对某一个用户群来群发一组短信。前面这些就是典型的应用场景&#xff0c;在 Abp 框…

【转】x.509证书在WCF中的应用(CS篇)

【转自】x.509证书在WCF中的应用(CS篇) 为什么要用x.509证书? WCF的服务端和客户端之间&#xff0c;如 果不作任何安全处理(即服务端的<security mode"None">)&#xff0c;则所有传输的消息将以明文方式满天飞&#xff0c;在internet/intranet环境下无疑是很不…

从概念到案例:初学者须知的十大机器学习算法

本文先为初学者介绍了必知的十大机器学习&#xff08;ML&#xff09;算法&#xff0c;并且我们通过一些图解和实例生动地解释这些基本机器学习的概念。我们希望本文能为理解机器学习基本算法提供简单易读的入门概念。 机器学习模型 在《哈佛商业评论》发表「数据科学家是 21 世…

测试Live Write的插件

1、文字竖排&#xff1a; 删除&#xff0c;只因首页显示时太占空间。2、酷表情&#xff1a;3、Rhapsody SongI am listening to Sad Songs And Waltzes by Cake . Rhapsody.

手把手教你用7行代码实现微信聊天机器人 -- Python wxpy

环境要求&#xff1a; Windows / Linux / Mac OS Python 3.4-3.6&#xff0c;以及 2.7 版本 wxpy安装 ## 使用国内源安装速度快 pip install -U wxpy -i "https://pypi.doubanio.com/simple/" 实例 让机器人与所有好友聊天 from wxpy import * # 实例化&#xff0c;并…

Dapr 已在塔架就位 将发射新一代微服务

微服务是云原生架构的核心&#xff0c;通常使用Kubernetes 来按需管理服务扩展。微软一直走在 Cloud Native Computing Foundation的 最前沿&#xff0c;并通过使用Kubernetes来支持其超大规模Azure和其混合云Azure Stack&#xff0c;微软对云原生的投资一部分来自其工具&#…

python 复制文件_10 行 Python 代码写 1 个 USB 病毒

(给Python开发者加星标&#xff0c;提升Python技能)转自&#xff1a; 知乎-DeepWeaver昨天在上厕所的时候突发奇想&#xff0c;当你把usb插进去的时候&#xff0c;能不能自动执行usb上的程序。查了一下&#xff0c;发现只有windows上可以&#xff0c;具体的大家也可以搜索(搜索…

html5中外描边怎么写,CSS3实现文字描边的2种方法(小结)

问题最近遇到一个需求&#xff0c;需要实现文字的描边效果&#xff0c;如下图解决方法一首先想到去看CSS3有没有什么属性可以实现&#xff0c;后来被我找到了text-stroke该属性是一个复合属性&#xff0c;可以设置文字宽度和文字描边颜色该属性使用很简单&#xff1a;text-stro…

混凝土墙开洞_满城混凝土柱子切割资质齐全

满城混凝土柱子切割资质齐全专业楼板切割开洞&#xff0c;钢筋混凝土墙开门&#xff0c;开窗&#xff0c;开方洞。混泥土承重墙新开门洞、开窗、通风管道开洞、专业开楼梯口&#xff0c;楼梯口加固&#xff0c;地下室开门洞&#xff0c;水泥墙开门加固、楼板加固、砖墙开门开窗…

马云害怕的事还是发生了

当前&#xff0c;余额宝的收益维持在4%左右不能突破&#xff0c;只能用作“钱包”放点零钱了。 放银行或者余额宝收益偏低&#xff0c;股票市场又处于震荡周期&#xff0c;期货市场等不是普通人进得去的&#xff0c;还不如直接买较高收益的互联网理财产品。 比如屡受政策利好…

cmosfixr插件怎么用_3dmax插件神器|怎么用3dmax插件神器去完成背景墙的效果图设计?...

又到3dmax插件神器的小课堂时间了&#xff01;小伙伴们还记得之前几张的知识点吗&#xff1f;如果不记得自己去温习&#xff0c;温故而知新哦&#xff01;如果学会了&#xff0c;下面学习3dmax插件神器小技巧的第四章建模篇的第4.16小节&#xff1a;怎么用3dmax插件神器去完成背…

不懂这25个名词,好意思说你懂大数据?

如果你刚接触大数据&#xff0c;你可能会觉得这个领域很难以理解&#xff0c;无从下手。近日&#xff0c;Ramesh Dontha在DataConomy上连发两篇文章&#xff0c;扼要而全面地介绍了关于大数据的75个核心术语&#xff0c;这不仅是大数据初学者很好的入门资料&#xff0c;对于高阶…

ab压力测试_Apache ab压力测试的知识点

Apache-ab是著名的Web服务器软件Apache附带的一个小工具&#xff0c;它可以模拟多个并发请求&#xff0c;测试服务器的最大承载压力。ab 是apachebench的缩写,ab命令会创建多个并发访问线程&#xff0c;模拟多个访问者同时对某一URL地址进行访问。它的测试目标是基于URL的&…

现代云原生设计理念

前文传送门什么是云原生&#xff1f;现代设计理念你会如何设计云原生应用程序&#xff1f;需要遵循哪些原则、模式和最佳实践&#xff1f;需要特别关注哪些底层/操作&#xff1f;十二要素应用程序目前被普遍认可的基于云的方法论是"十二要素应用程序"&#xff0c;它给…

NFS服务器架设篇

大家好&#xff0c;本周我们的课程是NFS服务器的架设。下面我们分几个部分来介绍NFS服务器。一、NFS简介NFS是分布式计算机系统的一部分&#xff0c;一般在用unix和类unix的系统上实现文件的传输。而且可以把NFS服务器共享的目录挂载到本地&#xff0c;使用cp&#xff0c;cd&am…