防止可淹沒應用程序的池溢出 大多數(shù) ADO.NET 數(shù)據提供程序使用連接池,以提高圍繞 Microsoft 斷開連接的 .NET 結構構建的應用程序的性能。應用程序首先打開一個連接(或從連接池獲得一個連接句柄),接著運行一個或多個查詢,然后處理行集,最后將連接釋放回連接池。如果沒有連接池,這些應用程序將花費許多額外時間來打開和關閉連接。 當您使用 ADO.NET 連接池來管理基于 Web 的應用程序和客戶端/服務器 Web 服務應用程序的連接時,您的客戶通常會獲得更快的連接和更好的總體性能。但是,當您的應用程序或 Web 站點上突然涌入了同時希望進行連接的大量客戶時,會發(fā)生什么事情呢?您的應用程序會“沉沒”,還是會“游泳”?就像救生員一樣,您需要仔細監(jiān)視連接池,以維護它的良好性能,并防止連接池發(fā)生溢出。我們首先探討連接池可能溢出的原因,然后討論如何編寫代碼或使用 Windows 性能監(jiān)視器來監(jiān)視連接池。 正如我于 2003 年 5 月發(fā)表的 Swimming in the .NET Connection Pool (InstantDoc ID 38356) 一文中討論的那樣,當您使用連接池時,您需要知道許多有關可伸縮性和性能的詳細信息。請記住,您需要監(jiān)視和管理兩個基本因素:每個池管理的連接數(shù)和連接池的數(shù)量。在一個有效的生產系統(tǒng)中,池的數(shù)量通常很少(1 到 10),而且,使用中的連接的總數(shù)也很少(少于 12 )有效的查詢只用不到一秒鐘的時間就可以完成,并斷開連接。因此,即使有數(shù)百個客戶同時訪問您的 Web 站點,相對較少的幾個連接常常足以處理整個負載。為了使您的應用程序有效地運行,您必須使連接資源處于自己的控制之下,并要監(jiān)視池的狀態(tài),這樣,在監(jiān)視池發(fā)生溢出以及您的客戶開始抱怨(或離開您的網站)之前您會收到某種警告。 為什么會發(fā)生連接池溢出? 參加電子郵件討論組的人常常抱怨應用程序是如何在測試中是“龍”而在形成為產品時就變成了“蟲”的。有時,他們會報告說,當連接了大約 100 個客戶端時,應用程序會停止或掛起。請記住,一個池中的默認連接數(shù)是 100。如果您嘗試從池中打開 100 個以上的連接,ADO.NET 會使應用程序的連接請求排隊等候,直到有空閑的連接。應用程序(及其用戶)將這種情況視為進入 Web 頁的延遲或視為應用程序死鎖。讓我們首先討論一下這個問題是如何產生的。 在 ADO.NET 中,SqlClient .NET 數(shù)據提供程序為您提供了兩種打開和管理連接的方法。首先,當您需要手工管理連接時,可以使用 DataReader 對象。利用這種方法,您的代碼將構造一個 SqlConnection 對象,設置 ConnectionString 屬性,然后使用 Open 方法來打開連接。當代碼完成 DataReader 后,您要在 SqlConnection 對象停止作用之前關閉 SqlConnection。要處理行集,您可以將 DataReader 傳遞到應用程序中的另一個例程,但仍然需要確保 DataReader 及其連接處于關閉狀態(tài)。如果您不關閉 SqlConnection,代碼會“泄漏”每個操作的連接,于是連接池對連接進行累積,最后便發(fā)生溢出。與 ADO 和 Visual Basic (VB) 6.0 中的情況不同,.NET ***回收器不會為您關閉 SqlConnection 并進行清理。我稍后要討論的 清單 1 顯示了如何打開連接和生成 DataReader 以從一個簡單的查詢返回行集,來向連接池施加壓力的。 您也可能在使用 DataAdapter 對象時遇到問題。DataAdapter Fill 和 Update 方法可自動打開
用不到五個連接來處理每天的幾十萬次點擊。 標注 A 中的例程創(chuàng)建 SqlConnection 對象和 SqlCommand 對象,設置 CommandText,并打開連接。然后,標注 B 中的代碼確定執(zhí)行 DataReader 時是否使用 CommandBehavior.CloseConnection,這取決于用戶在 Web 窗體上選擇了哪些 CheckBox 控件。 在標注 C 的代碼中,我指定是否將 DataReader 行集綁定到 DataGrid,或者是否在整個行集中進行循環(huán)。標注 C 的代碼測試當您到達通過 DataReader 從數(shù)據提供程序傳遞回來的行集的末尾時會發(fā)生什么事情。 現(xiàn)在,我使用標注 D 中的代碼來指定是手工關閉連接還是讓某個其他操作(例如,數(shù)據綁定)來完成這項工作。坦白地說,以手工方式關閉連接通常是最安全的,因此,您可以肯定連接不會被孤立。 如果代碼成功地運行到這一步,說明我已經成功地打開和關閉了 110 個連接。不過,如果出了問題,標注 E 的代碼中的異常處理程序會將異常(通常是 Timeout)作為 InvalidOperationException 捕獲,該異常是連接池已滿時 ADO.NET 的響應方式。 表 1 匯總了各個選項使例程成功運行或失敗的方式。請注意,如果您不設置 CommandBehavior.CloseConnection 選項,您的操作最終會失敗 — 即使在使用綁定控件的情況下也是如此。即使您使用該選項,但如果您沒有使用復雜的綁定控件,或者沒有手工關閉 SqlDataAdapter 或 SqlConnection,該進程仍然會失敗。 當我結束了這些示例應用程序的運行后,我已經生成了 1000 多個以上的池連接 — 所有連接均處于孤立狀態(tài)。雖然“SQL Server 用戶連接”計數(shù)為 0,但留下大約 40 個連接池。在我重新引導系統(tǒng)之前,孤立的池不會消失。 我用于此測試的示例應用程序包括使用 DataAdapter 來返回行的例程。除非您手工管理連接,否則,DataAdapter 將正確地打開和關閉 SqlConnection 對象,因此,您不太可能遇到孤立的池連接。不過,如果您的應用程序同時使用 DataReader 和 DataAdapter,您可能會發(fā)現(xiàn),如果某個連接與一個未關閉的 DataReader 相關聯(lián),則 DataAdapter 無法針對該連接運行查詢。 確定連接池何時達到最大連接數(shù) 正如我在 Swimming in the .NET Connection Pool 一文中討論的那樣,當連接池達到您通過 Max Pool Size ConnectionString 選項指定的最大連接數(shù)時,ADO.NET 將阻止任何隨后打開額外連接的嘗試。如果某個連接在您在 ConnectionTimeout 選項中指定的時間之前變?yōu)榭捎茫?SPAN lang=EN-US>.NET 數(shù)據提供程序將向您的應用程序傳遞一個指向該連接的指針,以便將控件返回給應用程序。不過,如果沒有及時釋放任何連接,連接請求將引發(fā) InvalidOperationException 異常。 現(xiàn)在您必須決定要采取的措施,我不建議您告訴用戶您已經用完了所有連接。有些應用程序會通知用戶系統(tǒng)正忙于幫助其他客戶,并建議用戶稍后進行訪問。其他應用程序則播放一段動畫,通知用戶系統(tǒng)尚未死鎖,而是正在忙于處理他們的請求。同時,您的代碼重新嘗試操作。在所有情況下,您應該 記錄這些故障,以便幫助診斷問題的癥結所在,并記錄您已經耗盡了資源。 監(jiān)視連接池 您已經打開和關閉了一個連接,現(xiàn)在您希望知道該連接是否仍然處于打開狀態(tài)。您可以使用幾種方法來確定有多少連接仍然處于打開狀態(tài),以及它們正在執(zhí)行何種操作:
|
|
來自: 悟靜 > 《.net和asp.net》