效能考量因素 (Entity Framework)
本主題說明 ADO.NET Entity Framework 的效能特性,並提供一些考量因素以協助提升 Entity Framework 應用程式的效能。
查詢執行的階段
為進一步了解 Entity Framework 中的查詢效能,了解針對概念模型執行查詢並傳回資料做為物件時發生的作業,對您有所幫助。 以下資料表說明這一系列的作業。
作業 | 相對成本 | 頻率 | 註解 |
---|---|---|---|
載入中繼資料 |
一般 |
在每個應用程式定義域中執行一次。 |
Entity Framework 使用的模型和對應中繼資料會載入至 MetadataWorkspace。 這個中繼資料會在全域中作快取,並在相同的應用程式定義域中,提供給其他 ObjectContext 執行個體使用。 |
開啟資料庫連接 |
一般1 |
需要時。 |
因為開啟的資料庫連接會消耗重要的資源,Entity Framework 只會在需要時開啟和關閉資料庫連接。 您也可以明確開啟連接。 如需詳細資訊,請參閱管理連接和交易 (Entity Framework)。 |
產生檢視 |
高 |
在每個應用程式定義域中執行一次。 (可以預先產生。) |
在 Entity Framework 可以針對概念模型執行查詢,或儲存變更至資料來源之前,Entity Framework 必須產生本地查詢檢視集,才能存取資料庫。 由於產生這些檢視的成本很高,您可以預先產生檢視,在設計階段就把這些檢視加入至專案。 如需詳細資訊,請參閱 HOW TO:預先產生檢視表來改善查詢效能。 |
準備查詢 |
一般2 |
針對每個唯一查詢執行一次。 |
包括組成查詢命令、根據模型和對應的中繼資料產生命令樹,以及定義傳回資料的形式等成本。 由於會對 Entity SQL 查詢命令作快取,後續執行相同查詢命令時可減少些許時間。 您也可以使用已編譯的 LINQ 查詢,降低稍後執行的成本。 如需詳細資訊,請參閱已編譯的查詢 (LINQ to Entities)。 如需 LINQ 查詢執行的一般資訊,請參閱 LINQ to Entities. |
執行查詢 |
低2 |
針對每個查詢執行一次。 |
使用 ADO.NET 資料提供者,針對資料來源執行命令的成本。 由於大部分的資料來源都會對查詢計畫作快取,後續執行相同查詢命令可能會減少些許時間。 |
載入和使用型別 |
低3 |
針對每個 ObjectContext 執行個體執行一次。 |
載入型別,並針對概念模型定義的型別進行驗證。 |
追蹤 |
低3 |
針對每個查詢傳回的物件執行一次。 4 |
如果查詢使用 NoTracking 合併選項,這個階段不會影響效能。 如果查詢使用 AppendOnly、PreserveChanges 或 OverwriteChanges 合併選項,會在 ObjectStateManager 中追蹤查詢結果。 針對查詢傳回的每一個已追蹤物件產生 EntityKey,並用來建立 ObjectStateManager 中的 ObjectStateEntry。 如果可以針對 EntityKey,找到現有的 ObjectStateEntry,則會回傳現有的物件。 如果使用 PreserveChanges 或 OverwriteChanges 選項,傳回前會先更新物件。 如需詳細資訊,請參閱識別解析、狀態管理和變更追蹤。 |
具體化物件 |
一般3 |
針對每個查詢傳回的物件執行一次。 4 |
讀取傳回之 DbDataReader 物件、建立物件和設定屬性值的程序,是根據 DbDataRecord 類別之每一個執行個體上的值。 如果物件已於 ObjectContext 中存在,且查詢使用 AppendOnly 或 PreserveChanges 合併選項,則這個階段不會影響效能。 如需詳細資訊,請參閱識別解析、狀態管理和變更追蹤。 |
1 當資料來源提供者實作連接共用,開啟連接的成本會經由共用而分散。 SQL Server 的 .NET 提供者支援連接共用。
2 成本會隨著增加的查詢複雜度而增加。
3 總成本會與查詢傳回的物件數呈比例增加。
4 EntityClient 查詢不需要這個額外的成本,因為 EntityClient 查詢傳回的是 EntityDataReader,而不是物件。 如需詳細資訊,請參閱 Entity Framework 的 EntityClient 提供者。
其他考量
下列其他考量可能會影響 Entity Framework 應用程式的效能。
查詢執行
由於查詢可能會耗用資源,請考量查詢在程式碼裡的執行點,以及在哪部電腦上執行。
延後執行與立即執行
建立 ObjectQuery 或 LINQ 查詢時,可能不會立即執行查詢。 查詢執行會延後,直到需要結果時才執行,例如在 foreach (C#) 或 For Each (Visual Basic) 列舉期間,或指定填滿 List 集合時。 呼叫 ObjectQuery 上的 Execute 方法,或呼叫會傳回單一查詢的 LINQ 方法 (例如 First 或 Any) 時,會立即開始執行查詢。 如需詳細資訊,請參閱物件查詢 (Entity Framework) 和查詢執行 (LINQ to Entities)。
LINQ 查詢的用戶端執行
雖然 LINQ 查詢的執行會發生在裝載資料來源的電腦上,一個 LINQ 查詢中的某些部分可以在用戶端電腦上進行評估。 如需詳細資訊,請參閱查詢執行 (LINQ to Entities) 的<存放區執行>一節。
查詢和對應複雜度
在實體模型中,個別查詢和對應的複雜度對於查詢的效能有重大影響。
對應複雜度
在概念模型中的實體以及儲存體模型中的資料表之間,擁有比簡單的一對一對應更複雜的模型,因此,所產生的命令比擁有一對一對應的模型更複雜。
查詢複雜度
在命令中需要大量聯結的查詢會針對資料來源進行執行,否則傳回的大量資料可能會以下列方式影響效能:
針對概念模型進行的查詢看似簡單,但可能會導致針對資料來源執行更複雜的查詢。 發生這個問題的原因是 Entity Framework 會將針對概念模型的查詢轉譯為針對資料來源的同等查詢。 當概念模型中的單一實體集對應至資料來源中一個以上的資料表,或當實體之間的關聯性對應至聯結資料表時,針對資料來源查詢執行的查詢命令可能需要一個以上的聯結。
注意: 請使用 ObjectQuery 的 ToTraceString 方法或 EntityCommand 類別,檢視針對資料來源而執行之指定查詢的命令。 如需詳細資訊,請參閱 HOW TO:檢視存放區命令 (Entity Framework)。 巢狀 Entity SQL 查詢可在伺服器上建立聯結,並傳回大量資料列。
下列是投影子句中巢狀查詢的範例:
SELECT c, (SELECT c, (SELECT c FROM AdventureWorksModel.Vendor AS c ) As Inner2 FROM AdventureWorksModel.JobCandidate AS c ) As Inner1 FROM AdventureWorksModel.EmployeeDepartmentHistory AS c
此外,這類查詢會使查詢管線產生單一查詢,並重複跨巢狀查詢的物件。 因此,單一資料行可能會重複多次。 在某些些資料庫上 (包括 SQL Server),這個工作會使 TempDB 資料表變得非常大,降低伺服器的效能。 您執行巢狀查詢時應特別注意。
如果用戶端正在執行耗用資源與結果集大小成正比的作業,任何傳回大量資料的查詢可能會使效能降低。 在這種情況下,您應該考慮依查詢限制傳回的資料量。 如需詳細資訊,請參閱 HOW TO:逐頁檢視查詢結果 (Entity Framework)。
由 Entity Framework 自動產生的任何命令,會比由資料庫開發人員明確撰寫的類似命令更為複雜。 如果您需要明確控制針對資料來源執行的命令,請考慮資料表值函式的對應或預存程序。
關聯性
為最佳化查詢效能,您必須定義實體 (作為實體模型中的關聯和資料來源中的邏輯關聯性) 之間的關聯性,。
查詢路徑
根據預設,執行 ObjectQuery 時,不會傳回相關物件 (即使物件表示關聯性本身)。 您可以利用下列三種方法中的任何一種方式載入相關物件:
在執行 ObjectQuery 之前設定查詢路徑。
在物件公開的導覽屬性上呼叫 Load 方法。
將 ObjectContext 上的 LazyLoadingEnabled 選項設定為 true。 請注意,這個動作會在您使用 Entity Data Model Designer 產生物件層程式碼時自動完成。 如需詳細資訊,請參閱Generated Code Overview。
考慮要使用哪一種方式時,請特別留意,在對資料庫要求的數目與單一查詢所傳回的資料量之間,必須有所取捨。 如需詳細資訊,請參閱載入相關的物件 (Entity Framework)。
使用查詢路徑
查詢路徑會定義查詢傳回的物件圖形。 定義查詢路徑時,只需針對資料庫進行單一要求,即可傳回此路徑定義的所有物件。 使用查詢路徑可能會使表面上簡單的物件查詢變成要針對資料來源執行複雜的命令。 發生這種情況,是因為必須進行一或多次聯結,才能在單一查詢中傳回相關物件。 在針對複雜實體模型 (例如具有繼承的實體或包含多對多關聯性的路徑) 的查詢中,這種複雜性會變得更大。
注意: |
---|
使用 ToTraceString 方法可查看將會由 ObjectQuery 產生的命令。 如需詳細資訊,請參閱 HOW TO:檢視存放區命令 (Entity Framework)。 |
如果查詢路徑包含太多相關物件,或是物件包含太多資料列資料,資料來源可能會無法完成查詢。 如果查詢需要超過資料來源能力的中繼暫時儲存體,就會發生這種情況。 發生這種情況時,請明確載入相關物件來降低資料來源查詢的複雜性。
明確載入相關物件
您可以呼叫傳回 EntityCollection 之導覽屬性上的 Load 方法,或是呼叫 EntityReference,以明確載入相關物件。 明確載入物件必須在每次呼叫 Load 時反覆存取資料庫一次。
注意: |
---|
在傳回的物件集合中執行迴圈時,如果呼叫 Load,例如使用 foreach 陳述式時 (Visual Basic 中的 For Each),資料來源特定的提供者必須支援單一連接上的多個作用中結果集。 若為 SQL Server 資料庫,您必須在提供者連接字串中指定 MultipleActiveResultSets = true 的值。
|
當實體上沒有 EntityCollection 或 EntityReference 屬性時,您也可以使用 LoadProperty 方法。 這在您使用 POCO 實體時很有用。
雖然明確載入相關物件將降低聯結數量和多餘資料量,Load 卻需要重複的資料庫連接,當明確載入大量物件時,這可能會耗用大量成本。
儲存變更
呼叫 ObjectContext 上的 SaveChanges 方法時,會針對內容中每一個已加入、已更新或已刪除的物件,產生另一個建立、更新或刪除的命令。 這些命令會在單一交易中的資料來源上執行。 若含有查詢,建立、更新和刪除作業的效能會依概念模型中對應的複雜度而有所不同。
分散式交易
在明確交易中,需要由分散式交易協調器 (DTC) 管理之資源的作業,會比不需要 DTC 的相似作業耗用更多成本。 提升至 DTC 會發生以下狀況:
包含針對 SQL Server 2000 資料庫或其他資料來源之作業的明確交易,永遠會將明確交易提升至 DTC。
當連接是由 Entity Framework 管理時,會執行包含針對 SQL Server 2005 之作業的明確交易。 發生這種情況,是因為每當單一交易內的連接關閉又重新開啟時,SQL Server 2005 會提升至 DTC,這是 Entity Framework 的預設行為。 使用 SQL Server 2008 就不會發生 DTC 提升。 若要在使用 SQL Server 2005 時防止這個問題發生,您必須明確開啟和關閉交易內的連接。 如需詳細資訊,請參閱管理連接和交易 (Entity Framework)。
當一個或多個作業在 System.Transactions 交易內執行時,會使用明確交易。 如需詳細資訊,請參閱管理連接和交易 (Entity Framework)。
提升效能的策略
您可以利用下列策略,改善 Entity Framework 中查詢的整體效能。
預先產生檢視
應用程式第一次執行查詢時,根據實體模型產生檢視會耗用大量成本。 請使用 EdmGen.exe 公用程式,預先產生做為 Visual Basic 或 C# 程式碼檔案的檢視表,在設計期間就可以加入至專案中。 您也可以使用文字範本轉換工具組來產生預先編譯的檢視表。 預先產生的檢視表會在執行階段進行驗證,以確保與指定之實體模型的目前版本一致。 如需詳細資訊,請參閱 HOW TO:預先產生檢視表來改善查詢效能 和在 Entity Framework 4 中使用預先編譯/預先產生的檢視表隔離效能 (英文)。
當您處理非常大的模型時,必須考量以下事項:
.NET 中繼資料格式會將給定二進位格式的使用者字串字元數目限制為 16,777,215 (0xFFFFFF)。 如果您要針對極大的模型產生檢視表,而且檢視表檔案到達這個大小限制,您會得到「沒有剩餘的邏輯空間可以用來建立更多使用者字串」編譯錯誤。 這個大小限制適用於所有 Managed 二進位檔。 如需詳細資訊,請參閱示範如何在處理大型和複雜模型時避免錯誤發生的部落格 (英文)。
考慮針對查詢使用 NoTracking 合併選項
追蹤在物件內容中傳回的物件是必要成本。 偵測物件的變更,並確保相同邏輯實體的多個要求能傳回相同物件執行個體,需要將物件附加至 ObjectContext 執行個體。 如果沒有更新或刪除物件的計畫,且不需要識別管理,則執行查詢時,請考慮使用 NoTracking 合併選項。
使用已編譯的 LINQ 查詢
當您的應用程式執行了 Entity Framework 中結構類似的查詢多次時,您可經常增加效能,其方式是編譯查詢一次,然後使用不同的參數執行查詢多次。 例如,應用程式可能必須擷取特定城市中的所有客戶;此城市是使用者在執行階段於表單中所指定。 LINQ to Entities 支援針對這個用途所編譯的查詢。 此查詢只會在第一次執行期間編譯一次。 不過,您之後無法變更在編譯階段中,針對查詢所設定的合併選項。
如需詳細資訊,請參閱已編譯的查詢 (LINQ to Entities)。
傳回正確的資料量
在某些情況下,Include 方法指定查詢路徑會比較快速,因為需要反覆存取資料庫的次數較少。 不過,在其他情況下,額外反覆存取資料庫以載入相關物件可能會比較快速,因為較簡單的查詢加上較少的聯結,可產生較少的資料重複。 因此,我們建議您測試各種不同的擷取相關物件方法。 如需詳細資訊,請參閱載入相關的物件 (Entity Framework)。
若要防止單一查詢傳回太多資料,請考慮將查詢結果分頁成多個可管理的群組。 如需詳細資訊,請參閱 HOW TO:逐頁檢視查詢結果 (Entity Framework)。
限制 ObjectContext 的範圍
在大多數的情況下,您應該在 using
陳述式 (Visual Basic 中的 Using…End Using
) 中,建立 ObjectContext 執行個體。 確保當程式碼存在陳述式區塊時,與物件內容關聯的資源會自動公開,這麼做可以提高效能。 不過,當控制項繫結至由物件內容管理的物件時,只要繫結是必要且為手動公開的,則應維護 ObjectContext 執行個體。 如需詳細資訊,請參閱管理連接和交易 (Entity Framework)。
考慮手動開啟資料庫連接
當應用程式執行一系列的物件查詢,或是經常呼叫 SaveChanges,持續針對資料來源執行建立、更新和刪除作業時,Entity Framework 必須不斷開啟和關閉與資料來源的連接。 在這些情況下,請考慮在這些連接開始時,手動開啟連接,然後當作業完成時,關閉或處置連接。 如需詳細資訊,請參閱管理連接和交易 (Entity Framework)。
效能資料
有些 Entity Framework 的效能資料會公布在下列 ADO.NET 小組部落格 (英文) 的文章中: