总之,由于C#提供了一套丰富的API,因此可以写出相当优雅的算法来处理Task对象,并对这些对象所表示的任务进行安排。对任务的用法理解的越透彻,写出来的异步代码越清晰。
这里简单说明两个常用的API:
WhenAll:会根据现有的一批任务创建出一项新的任务,只有当那批任务全部执行完毕时,这项新人物才能够完成。对Task.WhenAll所返回的新任务进行await操作会获得一份列表,早前的那些任务的执行结果就位于该列表中。
WhenAny:为了尽早的获得某个结果,可能启动多项任务,使得他们分别从不同的途径去获取该结果。只要其中有一项任务完成,你的目标就达成了,针对这项需求,可以考虑使用Task.WhenAny方法,并把自己所创建的那批任务传进去。对WhenAny方法所返回的Task对象进行await操作可以获取到一项任务,它指的就是这批任务中最先执行完毕的那项任务。
考虑实现任务的取消协议
异步任务的编程模型(也叫基于任务的异步编程模型)提供了标准的API,用来取消任务或者广播任务的执行进度。虽然这些API是可选的,但如果某项任务确实能够汇报其进度,或者能够予以取消,那就可以考虑用合适的办法来实现这些API。
针对需要取消的任务,我们可以通过CanclelationTokenSource对象来进行取消操作。这种对象是一种起到中介作用的对象。该对象处在有可能发出取消请求的客户代码与支持取消功能的那项操作之间。
如果正在执行的任务发现客户端想要取消该操作,那么它就会通过ThrowIfCanclellationRequested()方法抛出TaskCanclledException异常,庸医表示整个工作流程没有能够完全得到执行。
此外,返回值类型为void类型的异步方法不应该支持取消功能。
缓存泛型异步方法的返回值
可能你在进行异步编程的时候对异步方法设置的返回类型都是Task或者Task<T>,然而有些时候把返回值类型设为Task可能会影响性能。如果某个循环或某段代码需要频繁的运行,那么系统就有可能分配很多个Task对象,从而占用相当多的资源。好在C#提供了一种新的类型,叫做ValueTask<T>对象,他用起来比普通的Task更为高效。该类型是值类型,因此创建这种类型的对象时,不需要再分配额外的空间。这个好处使得我们可以多创建一些这样的对象,而不用担心它会像Task对象那样占据过多的资源。如果你的异步方法可以根据早前缓存起来的结果直接返回相应的值,那么尤其应该考虑把返回值类型设置为ValueTask<T>。
其次,ValueTask提供了一个能够接受Task参数的构造函数,这个构造函数会在其内部等候该Task的执行结果。
总结
今天分享的内容比较多,而且很多都比较难理解,不过确实是写出高性能异步方法所必须要掌握的技巧。由于时间较短,因此也没来得及通过代码进行讲述,所以需要有一定的基础才能看懂,不过还是希望对您有所帮助。










