基于任务的异步编程(Task,async,await)

Posted charlesmvp

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了基于任务的异步编程(Task,async,await)相关的知识,希望对你有一定的参考价值。

这节讲一下比较高级的异步编程用法Task,以及两个异步关键字async和await。

    Task是在C#5.0推出的语法,它是基于任务的异步编程语法,是对Thread的升级,也提供了很多API,先看一下Task怎么使用:

System.Threading.Tasks.Task.Run(() =>
  {
      Console.WriteLine("异步");
  });
  System.Threading.Tasks.Task aTask=new System.Threading.Tasks.Task(() =>
  {
      Console.WriteLine("异步");
  });
  aTask.Start();
  System.Threading.Tasks.Task.Factory.StartNew(() =>
  {
      Console.WriteLine("异步");
  });

Task.Run()可以直接异步运行一个方法,或者使用实例化Task传入委托的方式,通过start()进行启动,再或者使用Task.Factory.StartNew()直接启动。

 

async,await

    为了进一步介绍Task,需要先介绍两个异步有关的关键字async,await

    async用在方法的声明,await用于代码语句中。被async标记的方法,称作异步方法。但是,并非整个方法都是异步执行,代码中以await开头标记的代码,才是要真正异步执行的具体内容。这个关键字一般是配合Task来使用的,Task有泛型的形式,标识异步的返回值类型,通过Result()方法获取返回值。这段说明很难理解,下面看代码演示:

static void Main(string[] args)
{
    Console.WriteLine("新建一个异步任务");
    Task<int> task = new Program().GetValue();
?
    Console.WriteLine("正在计算结果....");
    Console.WriteLine($"运行结果为:{task.Result}");
    Console.WriteLine("任务完成....");
}
public async Task<int> GetValue()
{
    Console.WriteLine("即将开始进行计算...3");
    Console.WriteLine("即将开始进行计算...2");
    Console.WriteLine("即将开始进行计算...1");
    int a= await System.Threading.Tasks.Task.Run(() =>
    {
        int i = 0;
         for (; i < 10; i++)
         {
             Console.WriteLine(i);
         }
         return i;
     });
    Console.WriteLine("结果计算完成....");
    return a;
}

 运行结果为:

技术图片

从运行结果可以看出,程序运行到15行await处后,下一步就跳出了这个方法,回到第6行执行,这也是await的一个特性,异步执行,将主线程执行权交回,也就是说,从15行到25行是在后台线程中执行的,之前的执行都是同步的,之后的执行也是同步的,而且,主线程的脚步没有停下,直到遇到task.Result,Result里边存放着异步方法运行的返回值,运行到这,如果异步没有完成,就会阻塞当前线程,直到异步返回结果。

    另外说一点,之前在讲自定义中间件的时候,涉及到过这两个关键词,现在明白了这个用法,可以回去再看一下,应该会对中间件的访问流程有一个更清晰的理解。

 

    ContinueWith

    ContinueWith设置Task在执行完原有任务后,再继续执行此方法设置的方法,下面看代码:

task.ContinueWith((task) =>
  {
      Console.WriteLine("---------------"+a.Result);
  });

 这是其中的一个重载,接受一个Action<Task<T>>类型的委托,此处乍一看可能会不解,其实就是把当前执行任务的Task对象传进来了。这样的用法有什么好处呢,运行完了以后,可以直接取Task任务的返回值,不用阻塞线程,当然这是在返回值不是急需的情况下。

 

CancellationTokenSource

    CancellationTokenSource类用于终止一个任务,请先看一下代码:

static void Main(string[] args)
{
    CancellationTokenSource cancellationToken=new CancellationTokenSource();
?
    Task<int> task= new Program().GetValue(cancellationToken);
    cancellationToken.Cancel();
    Console.WriteLine("正在计算结果....");
?
    Console.WriteLine($"运行结果为:{task.Result}");
    Thread.Sleep(1000);
    Console.WriteLine("任务完成....");
}
public Task<int> GetValue(CancellationTokenSource cancellationToken)
{
    Task<int> a = System.Threading.Tasks.Task.Run(() =>
     {
         int i = 0;
         for (; i < 100; i++)
         {
             Console.WriteLine(i);
             Thread.Sleep(500);
         }
         return i;
     },cancellationToken.Token);
    Console.WriteLine("结果计算完成....");
    return a;
}

创建一个CancellationTokenSource对象,在Run任务的时候传入一个Token,就能调用Cancel()方法就能终止这个任务,运行结果为:技术图片

 可以看到报错了,这很正常,因为任务停止了,显然Result是没有值的

    最后注意一点,异步不是多线程,可以说异步是基于多线程,但是它们不是等于的关系。

 

这是我的公众号二维码,获取最新文章,请关注此号

技术图片

以上是关于基于任务的异步编程(Task,async,await)的主要内容,如果未能解决你的问题,请参考以下文章

基于任务的异步编程模式(TAP)

Task异步编程

任务Task系列之异步编程(async and await)

.Net Core WebAPI 基于Task的同步&异步编程

async & await 异步编程的一点巧方法

C#异步编程概念和使用