自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

C#.NET 拾遺補(bǔ)漏之理解 C# 中的異步流

網(wǎng)絡(luò) 通信技術(shù)
異步流(Asynchronous Stream)就是這兩種功能的結(jié)合體,它實(shí)現(xiàn)了以異步的方式生成和消費(fèi)一組數(shù)據(jù)系列的迭代器。

[[440225]]

本文轉(zhuǎn)載自微信公眾號(hào)「精致碼農(nóng)」,作者liamwang  。轉(zhuǎn)載本文請(qǐng)聯(lián)系精致碼農(nóng)公眾號(hào)。

在閱讀本文前,建議先閱讀本系列的上一篇文章『理解 yield 關(guān)鍵字』。我們知道,使用 C# 的 yield 關(guān)鍵字可以實(shí)現(xiàn)一個(gè)迭代器(Iterator),使用 async/await 關(guān)鍵字可以實(shí)現(xiàn)一個(gè)異步方法。異步流(Asynchronous Stream)就是這兩種功能的結(jié)合體,它實(shí)現(xiàn)了以異步的方式生成和消費(fèi)一組數(shù)據(jù)系列的迭代器。

異步流的支持主要建立在 C# 8 引入的兩個(gè)接口上:

  1. public interface IAsyncEnumerable<out T> 
  2.     IAsyncEnumerator<T> GetAsyncEnumerator (...); 
  3. public interface IAsyncEnumerator<out T>: IAsyncDisposable 
  4.     T Current { get; } 
  5.     ValueTask<bool> MoveNextAsync(); 

所以理解了上一篇我們講的 yield 關(guān)鍵字,就很容易理解異步流,它只是在原來的基礎(chǔ)上支持通過 yield return 返回異步得到的一系列結(jié)果值而已。從序列中獲取每個(gè)元素的行為(MoveNextAsync)是一個(gè)異步操作,元素是以零散的方式到達(dá),這就形成了所謂的“異步流”(比如視頻流中的數(shù)據(jù))。

這里 IAsyncEnumerator 接口的 ValueTask 是 Task 類型輕量化的封裝,它是結(jié)構(gòu)類型(值類型)。使用方式與 Task 相似,但它在同步完成任務(wù)或返回立即可用的結(jié)果時(shí)(這在列舉序列時(shí)會(huì)經(jīng)常發(fā)生),可以避免不必要的內(nèi)存開銷,比 Task 更高效。

在上一篇文章中的 Fibonacci 方法中,Thread.Sleep(1000) 用來模擬一個(gè)耗時(shí)操作,它是“同步”的:

  1. IEnumerable<int> Fibonacci(int count
  2.     int prev = 1; 
  3.     int curr = 1; 
  4.     for (int i = 0; i < count; i++) 
  5.     { 
  6.         yield return prev; 
  7.         Thread.Sleep(1000); 
  8.         int temp = prev + curr; 
  9.         prev = curr; 
  10.         curr = temp
  11.     } 

為了提高程序執(zhí)行效率,我們很有可能需要把 Thread.Sleep(1000) 改成“異步”的。如果使它生成異步的數(shù)據(jù)流,該怎么做呢?這就需要同時(shí)用到迭代器和異步方法了,也就是說方法中要同時(shí)使用 yield return 和 async/await 關(guān)鍵字。要支持這一特性,就要使用 IAsyncEnumerable 作為方法的返回類型。于是,前文的 Fibonacci 方法可以這樣改造:

  1. async IAsyncEnumerable<int> FibonacciAsync(int count
  2.     int prev = 1; 
  3.     int curr = 1; 
  4.     Random r = new(); 
  5.     for (int i = 0; i < count; i++) 
  6.     { 
  7.         yield return prev; 
  8.         await Task.Delay(1000); 
  9.         int temp = prev + curr; 
  10.         prev = curr; 
  11.         curr = temp
  12.     } 

不同的是,這個(gè)方法允許調(diào)用者以異步的方式消費(fèi)它生成的數(shù)字系列,換句話說就是使用 await foreach 來遍歷消費(fèi)這個(gè)方法的返回結(jié)果(IAsyncEnumerable),如下所示:

  1. await foreach (var n in FibonacciAsync(10)) 
  2.     Console.Write("{0} ", n); 

但,如果要在 LINQ 查詢語句中消費(fèi)異步流,需要先引入 System.Linq.Async NuGet 包,除此之外,使用方式?jīng)]有差別:

  1. IAsyncEnumerable<int> query = 
  2.     from i in FibonacciAsync(10) 
  3.     where i % 2 == 0 
  4.     select i * 10; 
  5.  
  6. await foreach (var number in query) 
  7.     Console.WriteLine(number); 

另外,在 ASP.NET Core 的 Action 方法中也支持返回異步流,如下面示例:

  1. [HttpGet] 
  2. public async IAsyncEnumerable<string> Get() 
  3.     using var dbContext = new BookContext(); 
  4.     await foreach (var title in dbContext.Books 
  5.         .Select(b => b.Title) 
  6.         .AsAsyncEnumerable()) 
  7.     yield return title; 

綜上,可以看到,異步流解決了零散數(shù)據(jù)系列的異步生成和消費(fèi)問題。要知道,在 C# 還沒有異步流時(shí),一組數(shù)據(jù)系列(IEnumerable)只能以整體異步的方式(Task

參考:

  1. 《C# 9.0 in a nutshell》 
  2. https://docs.microsoft.com/en-us/dotnet/csharp/whats-new/tutorials/generate-consume-asynchronous-stream 

 

責(zé)任編輯:武曉燕 來源: 精致碼農(nóng)
相關(guān)推薦

2024-09-23 12:35:49

2022-10-12 23:39:46

Java接口屬性

2022-10-11 09:33:04

Java異常Exception

2018-11-30 15:17:38

CPUCache緩存行

2010-03-30 08:36:26

Java框架StrutsSpring

2011-06-17 15:55:19

ArrayListC#

2009-08-25 13:53:20

C#.NET rege

2009-08-26 14:23:14

C#.Net Fram

2009-04-02 15:21:43

c#IDisposeFinalize

2010-07-20 08:50:00

autoreleaseObjective C

2009-08-13 10:35:55

C#.NET操作XML

2009-08-24 16:19:54

C#.NET綁定Off

2009-08-26 10:09:22

C#編碼規(guī)范

2009-09-01 18:26:23

C#工作流引擎

2024-05-11 07:13:33

C#Task編程

2024-06-25 08:33:48

2009-09-01 16:14:05

ArrayList與A

2021-01-19 05:30:55

C# 8異步流IEnumerable

2021-06-25 10:18:08

JavaScript Array.map 巧技拾遺

2021-09-13 07:00:01

C# .NET 緩存
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)