接上文 多线程编程学习笔记——async和await(一)

三、   对连续的异步任务使用await操作符

本示例学习如何阅读有多个await方法方法时,程序的实际流程是怎么样的,理解await的异步调用 。

1。示例代码如下。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Threading; namespace ThreadAsyncDemo
{ class Program
{
static void Main(string[] args)
{

Console.WriteLine(string.Format("----- 对连续的异步任务使用await操作符----"));

            Task t = AsyncWithTPL();
t.Wait();
t = AsyncWithAwait();
t.Wait();
Console.Read();
} static Task AsyncWithTPL()
{
var containerTask = new Task(() =>
{
Task<string> task1 = GetInfoAsync("TPL Task 1"); task1.ContinueWith(task =>
{
Console.WriteLine(task1.Result); Task<string> task2 = GetInfoAsync("TPL Task 2");
task2.ContinueWith(innerTask =>
Console.WriteLine(innerTask.Result), TaskContinuationOptions.NotOnFaulted | TaskContinuationOptions.AttachedToParent); task2.ContinueWith(innerTask =>
Console.WriteLine(innerTask.Exception.InnerException), TaskContinuationOptions.OnlyOnFaulted | TaskContinuationOptions.AttachedToParent); }, TaskContinuationOptions.NotOnFaulted | TaskContinuationOptions.AttachedToParent);
task1.ContinueWith(task =>
Console.WriteLine(task1.Exception.InnerException), TaskContinuationOptions.OnlyOnFaulted | TaskContinuationOptions.AttachedToParent);
}
);
containerTask.Start();
return containerTask;
}
async static Task AsyncWithAwait()
{
try
{
string result = await GetInfoAsync("Async Task 1");
Console.WriteLine(result);
result = await GetInfoAsync("Async Task 2");
Console.WriteLine(result);
}
catch (Exception ex)
{
Console.WriteLine(ex.Message);
}
} async static Task<string> GetInfoAsync(string name)
{
Console.WriteLine(string.Format("任务 {0} 开始。。。",name));
await Task.Delay(TimeSpan.FromSeconds());
if(name== "TPL Task 2")
throw new Exception(string.Format("{0} 抛出异常信息!", name));
return string.Format(" Task {0} 正在运行在线程 ID={1}上。这个工作线程是否是线程池中的线程:{2}", name,
Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread);
}
}
}

2.程序运行结果,如下图。

首先,程序运行时会执行两个异步操作。

  1. AsyncWithAwait方法,它看起来跟平常的代码一样,唯一不同之处就是使用了两个await声明。最重要的一点是代码的执行顺序依然是顺序执行的,调用AsyncWithAwait任务的代码只有等 之前的的任务完成 后才会开始执行。当阅读代码时,程序流很清晰,可以看到什么先执行,什么后执行,但是程序如何异步执行呢?首先,它不是一直是异步执行的,只有当使用await,且一个任务已经完成,我们会异步地得 到这个任务的结果。否则 ,当在代码中看到await声明时,通过的行为是代码执行到await代码行时将立即返回,并且剩下的代码将会在一个后续操作任务中运行。因此等待操作结果时并没有阻塞程序执行,这是一个异步调用 。当AsyncWithAwait方法中的代码执行时,除了在main方法中调用 t.wait外,我们可以执行任务其他任务。但是主线程必须等待直到所有的异步操作完成 ,否则 主线程完成后所有运行异步操作的后台线程会停止运行。
  2. AsyncWithTPL方法模仿了AsyncWithAwait的程序流。我们需要一个容器任务来处理所有相互依赖的任务。然后启动主任务,给其加了一组后续操作。当这个任务完成 后,会打印其结果。然后又启动了一个任务,在这个任务完成后会依次运行更多的后续操作。为了测试对异常的处理,当运行第二个任务时故意抛出了一个异常。并打印出异常信息。这组后续操作创建了与第一个方法中一样的程序流。

二、   对并行执行的异步任务使用await操作符

本示例学习如何使用await来并行地运行异步任务,而不采用常规使用顺序执行方式。

1.示例代码如下。

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using System.Threading; namespace ThreadAsyncDemo
{ class Program
{ static void Main(string[] args)
{
Console.WriteLine(string.Format("----- 对并行执行的异步任务使用await操作符----"));
Task t = AsyncProcess();
t.Wait();
Console.Read(); } async static Task AsyncProcess()
{ Task<string> task1 = GetInfoAsync("Task 1",); Task<string> task2 = GetInfoAsync("Task 2",);
string[] results = await Task.WhenAll(task1, task2);
foreach (var item in results)
{
Console.WriteLine(item);
}
} async static Task<string> GetInfoAsync(string name,int second)
{
Console.WriteLine(string.Format("任务 {0} 开始。。。",name)); await Task.Delay(TimeSpan.FromSeconds(second));
//await Task.Run(() => Thread.Sleep(TimeSpan.FromSeconds(second));
return string.Format(" Task {0} 正在运行在线程 ID={1}上。这个工作线程是否是线程池中的线程:{2}", name,
Thread.CurrentThread.ManagedThreadId, Thread.CurrentThread.IsThreadPoolThread);
}
}
}

2.程序运行结果,如下图。

在AsyncProcess方法中定义了两个异步任务task1,task2,分别运行3秒和5秒。然后使用Task.WhenAll辅助方式得到了两个任务的运行结果,而且只有在所有底层任务完成之后,才会运行。之后我们等待这个组合任务的结果。5秒后,我们获取所有结果,说明了这些任务是同时运行的。

我们仔细观察会发现,这两个任务假托是被线程池中的同一个工作线程执行的。当我们并行运行任务时怎么可能发生这样的情况呢?我们注释掉GetIntroAsync方法中的await Task.Delay代码 行,并解除对await task.run代码行的注释,然后再次运行程序。

我们会看到这种情况下,两个任务会被不同的工作线程执行。不同之处是task.delay在幕后使用了一个计时器,过程如下:从线程池中获取工作线程,它将等待task.delay方法返回结果。然后task.delay方法启动计时器并指定一段代码,这段代码会在计时器时间到了task.delay方法中指定的秒数后被 调用 ,之后,立即将工作线程返回到线程池中。当计时器事件运行时,又从线程池中任意获取一个可用的工作线程(可能跟上次运行的是同一个工作线程),并运行计时器指定的代码。

当使用task.run方法时,从线程池中获取了一个工作线程并将其阻塞几秒,然后获取了第二个工作线程也将其阻塞。在这种情况下,我们消费了两个工作线程,但是什么事情也没做。如下图。比较两者的不同之处,在于两个task运行在不同的线程上。

多线程编程学习笔记——async和await(二)的更多相关文章

  1. 多线程编程学习笔记——async和await(三)

    接上文 多线程编程学习笔记——async和await(一) 接上文 多线程编程学习笔记——async和await(二) 五.   处理异步操作中的异常 本示例学习如何在异步函数中处理异常,学习如何对多 ...

  2. 多线程编程学习笔记——async和await(一)

    接上文 多线程编程学习笔记——任务并行库(一) 接上文 多线程编程学习笔记——任务并行库(二) 接上文 多线程编程学习笔记——任务并行库(三) 接上文 多线程编程学习笔记——任务并行库(四) 通过前面 ...

  3. 多线程编程学习笔记——编写一个异步的HTTP服务器和客户端

    接上文 多线程编程学习笔记——使用异步IO 二.   编写一个异步的HTTP服务器和客户端 本节展示了如何编写一个简单的异步HTTP服务器. 1.程序代码如下. using System; using ...

  4. 多线程编程学习笔记——使用异步IO(一)

    接上文 多线程编程学习笔记——使用并发集合(一) 接上文 多线程编程学习笔记——使用并发集合(二) 接上文 多线程编程学习笔记——使用并发集合(三) 假设以下场景,如果在客户端运行程序,最的事情之一是 ...

  5. 多线程编程学习笔记——异步调用WCF服务

    接上文 多线程编程学习笔记——使用异步IO 接上文 多线程编程学习笔记——编写一个异步的HTTP服务器和客户端 接上文 多线程编程学习笔记——异步操作数据库 本示例描述了如何创建一个WCF服务,并宿主 ...

  6. 多线程编程学习笔记——使用异步IO

    接上文 多线程编程学习笔记——使用并发集合(一) 接上文 多线程编程学习笔记——使用并发集合(二) 接上文 多线程编程学习笔记——使用并发集合(三) 假设以下场景,如果在客户端运行程序,最的事情之一是 ...

  7. Java多线程编程(学习笔记)

    一.说明 周末抽空重新学习了下多线程,为了方便以后查阅,写下学习笔记. 有效利用多线程的关键是理解程序是并发执行而不是串行执行的.例如:程序中有两个子系统需要并发执行,这时候需要利用多线程编程. 通过 ...

  8. [Java123] JDBC and Multi-Threading 多线程编程学习笔记

    项目实际需求:DB交互使用多线程实现 多线程编程基础:1.5  :( (假设总分10) 计划一个半月从头学习梳理Java多线程编程基础以及Oracle数据库交互相关的多线程实现 学习如何通过代码去验证 ...

  9. python学习笔记 async and await

    用asyncio提供的@asyncio.coroutine可以把一个generator标记为coroutine类型,然后在coroutine内部用yield from调用另一个coroutine实现异 ...

随机推荐

  1. http 500错误怎么解决方法

    出现500错误的原因是很多的,一般来说,如果程序出错,那么在浏览器内会返回给用户一个友好的错误提示,统一称之为服务器500错误. 解决的方法就是您必须在http中能够正确的获得错误信息,方法为:请打开 ...

  2. js实现查找字符串中最多的字符的个数

    用hash table实现.key是字符,value是字符个数. var hashTable={}; var str="fjsdeiuwidshjfhjsksghfjhsjjskalsk&q ...

  3. jsp静态与动态包含的区别和联系

    1. <%@ include file=” ”%>是指令元素.<jsp:include page=” ”/>是行为元素 2. 最终编译成java文件的数目不同. * 静态包含在 ...

  4. linux-cp

    cp 更改时间: 2017-10-26 - 21:00:54 cp:用来复制文件或者目录的命令,当源文件与目标文件名字相同的时候,当cp 没有参数,源文件会覆盖目标文件 参数 -p:保留源文件或者目录 ...

  5. SpringMVC知识一锅烩

    Spring简介 SpringMVC和Struts2一样都是属于表现层的框架,将前段发出的请求分发给对应的后端处理器即Controller 处理流程 用户请求被前端控制前拦截,然后根据对应的拦截路径去 ...

  6. selenim之ActionChains(一)

    大家好,来介绍下,今天要分享的是小编学ActionChains的经验. 先来说一下今天要用到的方法: click(element=null)                               ...

  7. Java中的Redis应用

    1.配置redis集群   1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 <?xml version="1.0" encoding ...

  8. 阿里云RDS SQL Server 2008 R2 使用本地SQL备份文件还原全过程

    最近公司准备全面转向阿里云,写了好几个方案,最终决定购买一台ECS和一台RDS搭配使用.开始对阿里的RDS产品陌生,加上公司的数据库文件近20G,诸多担心,生怕产品买来了不能用,给公司造成损失.后来联 ...

  9. 无限分级Repeater递归实现:读取一次数据库,使用LINQ2SQL技术,支持排序&amp;显示隐藏

    预览效果图: Selenium 数据库结构: id(int)    classname(string)   parentid(int) sort(int用于显示与排序) 1 家居 0 1 2 家电 0 ...

  10. Qemu 简述

    Qemu 架构 Qemu 是纯软件实现的虚拟化模拟器,几乎可以模拟任何硬件设备,我们最熟悉的就是能够模拟一台能够独立运行操作系统的虚拟机,虚拟机认为自己和硬件打交道,但其实是和 Qemu 模拟出来的硬 ...