Entity Framework非同步查詢和保存

来源:http://www.cnblogs.com/from1991/archive/2016/04/23/5425263.html
-Advertisement-
Play Games

EF6開始提供了通過async和await關鍵字實現非同步查詢和保存的支持(.net 4.5及更高版本)。雖然不是所有的操作都能從非同步中獲益,但是耗時的操作、網路或IO密集型任務中,使用非同步可以提升客戶端性能和增強伺服器的擴展性。 本文將覆蓋一下主題: 實例演練非同步操作 創建模型 創建同步程式 改為異 ...


EF6開始提供了通過async和await關鍵字實現非同步查詢和保存的支持(.net 4.5及更高版本)。雖然不是所有的操作都能從非同步中獲益,但是耗時的操作、網路或IO密集型任務中,使用非同步可以提升客戶端性能和增強伺服器的擴展性。

本文將覆蓋一下主題:

  • 實例演練非同步操作
  • 創建模型
  • 創建同步程式
  • 改為非同步操作

實例演練非同步操作

下麵演練將通過對比,很容易的觀察非同步操作和同步操作,該演練目的不是說明何時才是非同步操作的關鍵場景。

創建模型

下麵使用CodeFirst的流程創建模型並生成資料庫,不過非同步方法可以很好的工作於所有EF模型,包括EF設計器生成的模型。

創建一個控制台應用程式AsyncDemo。

添加EntityFramework NuGet包到項目中。

添加Model.cs到項目中,代碼如下:

 1 using System.Collections.Generic; 
 2 using System.Data.Entity; 
 3  
 4 namespace AsyncDemo 
 5 { 
 6     public class BloggingContext : DbContext 
 7     { 
 8         public DbSet<Blog> Blogs { get; set; } 
 9         public DbSet<Post> Posts { get; set; } 
10     } 
11  
12     public class Blog 
13     { 
14         public int BlogId { get; set; } 
15         public string Name { get; set; } 
16  
17         public virtual List<Post> Posts { get; set; } 
18     } 
19  
20     public class Post 
21     { 
22         public int PostId { get; set; } 
23         public string Title { get; set; } 
24         public string Content { get; set; } 
25  
26         public int BlogId { get; set; } 
27         public virtual Blog Blog { get; set; } 
28     } 
29 }

 

創建同步程式

有了EF模型,下麵通過代碼模擬資料庫存取。

 1 using System; 
 2 using System.Linq; 
 3  
 4 namespace AsyncDemo 
 5 { 
 6     class Program 
 7     { 
 8         static void Main(string[] args) 
 9         { 
10             PerformDatabaseOperations(); 
11  
12             Console.WriteLine(); 
13             Console.WriteLine("Quote of the day"); 
14             Console.WriteLine(" Don't worry about the world coming to an end today... "); 
15             Console.WriteLine(" It's already tomorrow in Australia."); 
16  
17             Console.WriteLine(); 
18             Console.WriteLine("Press any key to exit..."); 
19             Console.ReadKey(); 
20         } 
21  
22         public static void PerformDatabaseOperations() 
23         { 
24             using (var db = new BloggingContext()) 
25             { 
26                 // Create a new blog and save it 
27                 db.Blogs.Add(new Blog 
28                 { 
29                     Name = "Test Blog #" + (db.Blogs.Count() + 1) 
30                 }); 
31                 db.SaveChanges(); 
32  
33                 // Query for all blogs ordered by name 
34                 var blogs = (from b in db.Blogs 
35                             orderby b.Name 
36                             select b).ToList(); 
37  
38                 // Write all blogs out to Console 
39                 Console.WriteLine(); 
40                 Console.WriteLine("All blogs:"); 
41                 foreach (var blog in blogs) 
42                 { 
43                     Console.WriteLine(" " + blog.Name); 
44                 } 
45             } 
46         } 
47     } 
48 }

上面代碼通過調用PerformDatabaseOperations() 保存一個Blog對象到資料庫中,然後從資料庫中檢索所有Blog,並顯示到控制台,然後顯示一行文本”Quote of the day“。

由於上面程式是同步執行的,所有可以觀察到程式按下麵流程執行:

  1. SaveChanges保存Blog對象到資料庫中。
  2. SaveChanges完成。
  3. 發送查詢Blog請求到資料庫。
  4. 查詢返回結果,並寫入控制台。
  5. 顯示文本“Quote of the day”到控制台。

 

改造為非同步操作

對上面程式加以修改,使用async和await關鍵字實現非同步操作。

 1 using System; 
 2 using System.Data.Entity; 
 3 using System.Linq; 
 4 using System.Threading.Tasks; 
 5  
 6 namespace AsyncDemo 
 7 { 
 8     class Program 
 9     { 
10         static void Main(string[] args) 
11         { 
12             var task = PerformDatabaseOperations(); 
13  
14             Console.WriteLine("Quote of the day"); 
15             Console.WriteLine(" Don't worry about the world coming to an end today... "); 
16             Console.WriteLine(" It's already tomorrow in Australia."); 
17  
18             task.Wait(); 
19  
20             Console.WriteLine(); 
21             Console.WriteLine("Press any key to exit..."); 
22             Console.ReadKey(); 
23         } 
24  
25         public static async Task PerformDatabaseOperations() 
26         { 
27             using (var db = new BloggingContext()) 
28             { 
29                 // Create a new blog and save it 
30                 db.Blogs.Add(new Blog 
31                 { 
32                     Name = "Test Blog #" + (db.Blogs.Count() + 1) 
33                 }); 
34                 Console.WriteLine("Calling SaveChanges."); 
35                 await db.SaveChangesAsync(); 
36                 Console.WriteLine("SaveChanges completed."); 
37  
38                 // Query for all blogs ordered by name 
39                 Console.WriteLine("Executing query."); 
40                 var blogs = await (from b in db.Blogs 
41                             orderby b.Name 
42                             select b).ToListAsync(); 
43  
44                 // Write all blogs out to Console 
45                 Console.WriteLine("Query completed with following results:"); 
46                 foreach (var blog in blogs) 
47                 { 
48                     Console.WriteLine(" - " + blog.Name); 
49                 } 
50             } 
51         } 
52     } 
53 }

現在程式變為非同步執行,可以觀察到非同步執行順序為:

  1. 發送SaveChanges請求到資料庫。
  2. 該請求發送給資料庫時,當前線程不在占用CPU時間,從方法PerformDatabaseOperations中返回(雖然該方法還沒有執行完成),控制權返回給主線程執行。
  3. 顯示字元串“Quote of the day ”到控制台。
  4. SaveChanges完成。
  5. 發起查詢Blogs請求到資料庫。
  6. 查詢完成返回結果,並顯示到控制台。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、禁止root用戶遠程登錄: 將 permitRootLogin 後面的值改成 no 如下圖: 然後再重啟sshd服務即可,如下: 二、更改ssh遠程登錄埠: 去掉#後,將 Port 值改成 18206,如下圖: 重啟sshd服務即可. 三、限制ssh登錄用戶數量: 去掉#後,將 MaxSess ...
  • ngx_http_upstream_module用於將多個伺服器定義成伺服器組,而由proxy_pass,fastcgi_pass等指令引用 (1)upstream name {...} 定義一個後端伺服器組,name為組名,只能用於http上下文中 (2) server address [para ...
  • sudo apt-get install shutter 然後配合系統快捷鍵,我定義的和qq的截屏一樣的。用著感覺很舒服。 ...
  • Debian系統的普通用戶需要安裝軟體時,往往會收到“Permission denied”的提示,這時候需要root許可權。那麼如何在不登陸超級管理員賬戶的前提下擁有root許可權呢?對於大多數Linux系統來說,我們可以通過“sudo”命令來獲取root許可權,或者通過“su”登陸超級管理員賬戶來進行各 ...
  • C#泛型是一種高復用性、安全和高效的技術,通過類型參數可以將參數的聲明、實現推遲到客戶代碼中。但是這種延遲卻降低了類型參數在泛型定義中的可操作性。例如資源釋放。 但是如果T實現了IDisposable介面,則上面代碼可能存在資源泄露的風險。但是由於不知道T是否實現了IDisposable介面,所以不 ...
  • 菜鳥要飛系列目錄 1.(菜鳥要飛系列)一,基於Asp.Net MVC5的後臺管理系統(前言) 2.(菜鳥要飛系列)二,基於Asp.Net MVC5的後臺管理系統(實現登陸功能) 3.(菜鳥要飛系列)三,基於Asp.Net MVC5的後臺管理系統(用戶的增刪改查功能) 4.(菜鳥要飛系列)四,基於As ...
  • 最近由於伺服器變更為Linux系統.MsSql for Linux什麼時候出來到生產環境使用還是要很長時間的.於是考慮使用Mysql資料庫,ORM使用EF.於是先踩下坑順便記錄一下,有需要的tx可以參考下.當你考慮使用EF連接Mysql的時候肯定是已經在網上搜了一堆教程.網上教程基本都是使用控制台做 ...
  • 現在在Windows下的應用程式開發,VS.Net占據了絕大多數的份額。因此很多以前搞VC++開發的人都轉向用更強大的VS.Net。在這種情況下,有很多開發人員就面臨瞭如何在C#中使用C++開發好的類的問題。下麵就用一個完整的實例來詳細說明怎樣用托管C++封裝一個C++類以提供給C#使用。 比如,現 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...