微軟MVP為您詳述ADO.NET連線池
對於ADO.NET連線池,大家不會陌生。不過多次用過ADO.NET連線池的各位.NET程式設計師,卻不一定深入瞭解它,具體原因是:
ADO.NET中提供了連線池的功能,多數開發人員很少設定它,因為它是預設的。
關閉連線池也很簡單,在連線字串如下:
Data Source=(local);Initial Catalog=AdventureWorks;Integrated Security=SSPI;Pooling=False;
但連線池的本質是什麼樣的呢?
用Reflector,開啟System.Data.SqlClient.SqlConnection的ConnectionString屬性的設定值的方法,如下:
- private void ConnectionString_Set(string value)
- {
- DbConnectionOptions userConnectionOptions = null;
- DbConnectionPoolGroup group = this.ConnectionFactory.GetConnectionPoolGroup(value, null, ref userConnectionOptions);
- DbConnectionInternal innerConnection = this.InnerConnection;
- bool allowSetConnectionString = innerConnection.AllowSetConnectionString;
- if (allowSetConnectionString)
- {
- allowSetConnectionString= this.SetInnerConnectionFrom(DbConnectionClosedBusy.SingletonInstance, innerConnection);
- if (allowSetConnectionString)
- {
- this._userConnectionOptions = userConnectionOptions;
- this._poolGroup = group;
- this._innerConnection = DbConnectionClosedNeverOpened.SingletonInstance;
- }
- }
- if (!allowSetConnectionString)
- {
- throw ADP.OpenConnectionPropertySet("ConnectionString", innerConnection.State);
- }
- if (Bid.TraceOn)
- {
- string str = (userConnectionOptions != null) ? userConnectionOptions.UsersConnectionStringForTrace() : "";
- Bid.Trace("<prov.DbConnectionHelper.ConnectionString_Set|API> %d#, '%ls'\n", this.ObjectID, str);
- }
- }
再連線 到紅色的GetConnectionPoolGroup方法,如下程式碼
- internal DbConnectionPoolGroup GetConnectionPoolGroup(string connectionString, DbConnectionPoolGroupOptions poolOptions,
- ref DbConnectionOptions userConnectionOptions)
- {
- DbConnectionPoolGroup group;
- if (ADP.IsEmpty(connectionString))
- {
- return null;
- }
- if (!this._connectionPoolGroups.TryGetValue(connectionString, out group) || (group.IsDisabled && (group.PoolGroupOptions != null)))
- {
- DbConnectionOptions options = this.CreateConnectionOptions(connectionString, userConnectionOptions);
- if (options == null)
- {
- throw ADP.InternalConnectionError(ADP.ConnectionError.ConnectionOptionsMissing);
- }
- string str = connectionString;
- if (userConnectionOptions == null)
- {
- userConnectionOptions = options;
- str = options.Expand();
- if (str != connectionString)
- {
- return this.GetConnectionPoolGroup(str, null, ref userConnectionOptions);
- }
- }
- if ((poolOptions == null) && ADP.IsWindowsNT)
- {
- if (group != null)
- {
- poolOptions = group.PoolGroupOptions;
- }
- else
- {
- poolOptions = this.CreateConnectionPoolGroupOptions(options);
- }
- }
- DbConnectionPoolGroup group2 = new DbConnectionPoolGroup(options, poolOptions) {
- ProviderInfo = this.CreateConnectionPoolGroupProviderInfo(options)
- };
- lock (this)
- {
- Dictionary<string, DbConnectionPoolGroup> dictionary = this._connectionPoolGroups;
- if (!dictionary.TryGetValue(str, out group))
- {
- Dictionary<string, DbConnectionPoolGroup> dictionary2 = new Dictionary<string, DbConnectionPoolGroup>(1 + dictionary.Count);
- foreach (KeyValuePair<string, DbConnectionPoolGroup> pair in dictionary)
- {
- dictionary2.Add(pair.Key, pair.Value);
- }
- dictionary2.Add(str, group2); this.PerformanceCounters.NumberOfActiveConnectionPoolGroups.Increment();
- group = group2;
- this._connectionPoolGroups = dictionary2;
- }
- return group;
- }
- }
- if (userConnectionOptions == null)
- {
- userConnectionOptions = group.ConnectionOptions;
- }
- return group;
- }
TryGetValue是判斷是否存在連線字串為connectionString的連線,存在返回到group,不存在就呼叫CreateConnectionOptions建立一個DbConnectionOptions,最後用
- lock (this)
- {
- Dictionary<string, DbConnectionPoolGroup> dictionary = this._connectionPoolGroups;
- if (!dictionary.TryGetValue(str, out group))
- {
- Dictionary<string, DbConnectionPoolGroup> dictionary2 = new Dictionary<string, DbConnectionPoolGroup>(1 + dictionary.Count);
- foreach (KeyValuePair<string, DbConnectionPoolGroup> pair in dictionary)
- {
- dictionary2.Add(pair.Key, pair.Value);
- }
- dictionary2.Add(str, group2); this.PerformanceCounters.NumberOfActiveConnectionPoolGroups.Increment();
- group = group2;
- this._connectionPoolGroups = dictionary2;
- }
- return group;
- }
這段程式碼放到連線池中,在這裡,可能顯示的看到,ado.NET的連線池實質上是一個Dictionary
所謂的連線池,就是一個與連線物件Connection相關的集合,這不只是簡單的集合,而是有一定的機制在內部。我們做開發時,可能建立Connection連線物件,關閉連線物件,有時候還呼叫Dispose來釋放連線。下次再用時,便重新例項化一個連線。但在池中的連線不隨連線物件的Close或Dispose而釋放。如果下次重新建立連線,連線字串與前一次完全一模一樣,則連線池就會把上次可用的連線物件賦給連線去用。如果兩個連線字串有一點不一樣,即使在某一個地方多一個空格,連線池也不會以為是相同的連線,這點微軟可能在內部只直接去比較兩個字串了,而不是比較連線資料庫字串的鍵值互相匹配。
連線池的好處就是保留連線物件,防止下次重頭再來例項化一個連線物件。
- string constr1 = "Data Source=(local);Initial Catalog=AdventureWorks;Integrated Security=SSPI;";
- string constr2 = "Data Source=(local);Initial Catalog=Pubs;Integrated Security=SSPI;";
- string AssMark = "System.Data,Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089";
- Assembly ass = Assembly.Load(AssMark);
- Type SqlConType = null;
- foreach (Type conType in ass.GetExportedTypes())
- {
- Console.WriteLine(conType .ToString ());
- if ("System.Data.SqlClient.SqlConnection" == conType.ToString())
- {
- SqlConType = conType;
- }
- }
- if (SqlConType != null)
- {
- Type[] types1 = new Type[0];
- ConstructorInfo constructorInfoObj1 = SqlConType.GetConstructor(
- BindingFlags.Instance | BindingFlags.Public, null,
- CallingConventions.HasThis, types1, null);
- SqlConnection con1 = (SqlConnection)constructorInfoObj1.Invoke(null);
- con1.ConnectionString = constr1;
- SqlConnection con2 = (SqlConnection)constructorInfoObj1.Invoke(null);
- con2.ConnectionString = constr2;
- PropertyInfo PI = SqlConType.GetProperty("PoolGroup", BindingFlags.Instance | BindingFlags.NonPublic);
- object poolGroup1 = PI.GetValue(con1, null);
- object poolGroup2 = PI.GetValue(con2, null);
- }
(說明:可能找到結果後覺得非常簡單,但怎麼找到結果的,卻是費了很大勁,幾乎是5個小時,所以相把找到結果的過程簡單說一下:
一開始用Reflector發現SqlConnection中有一個PoolGroup的屬性,於是就想在執行時候比較兩個SqlConnection物件的這個屬性,但由於這個屬性是的訪問修飾符是internal的,不能直接訪問,只有用反射,程式碼(是經過優化的)如下:
然後在倒數第一行設定斷點,為比較poolGroup1和poolGroup2的不同,結果發現,當連線字串一樣時,這兩個物件的_objectID相同,字串有一點不同就會不同,這點說明連線池中是用字串本身比較的,而不是字串中鍵值對進行比較。同還發現當con1和con2的ConnectionString不賦值時這兩個物件都是null,由此說明關鍵是ConnectionString賦值上,所以才開始用Reflector檢視這個屬性的賦值方法,才有上面的程式碼。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/12639172/viewspace-630624/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- ADO.NET連線池寫法
- 連線池 sql server ado.net 使用SQLServer
- 連線池和連線數詳解
- Ado.net中SQLServer資料庫連線池(轉)SQLServer資料庫
- ADO.NET入門教程之資料庫連線池資料庫
- ADO.NET連線字串字串
- Java Hibernate 之連線池詳解Java
- 連線池
- 資料庫連線池技術詳解資料庫
- ADO.NET連線資料庫資料庫
- ado.net 連線資料庫資料庫
- Go連線池Go
- HTTP連線池HTTP
- django連線池Django
- Http持久連線與HttpClient連線池HTTPclient
- 資料庫連線池的問題,連線池物件為靜態變數有問題麼資料庫物件變數
- 為vert x框架新增druid連線池支援框架UI
- ElasticSearch連線池建立Elasticsearch
- 自定義連線池
- golang tcp連線池GolangTCP
- Oracle 連線池配置Oracle
- oracle occi 連線池Oracle
- Resin 配置連線池
- [尋]連線池例子
- proxool連線池 配置
- 連線池 druidUI
- GaussDB(DWS)叢集通訊:詳解pooler連線池
- java連線池解決連線中斷Java
- proxool連線池如何使用SSL方式連線?
- 詳解 Tomcat 的連線數與執行緒池Tomcat執行緒
- Golang SQL連線池梳理GolangSQL
- go 語言連線池Go
- DBCP連線池原理分析
- Hibernate連線池配置
- Tomcat連線池使用Tomcat
- 資料庫連線池資料庫
- chrome瀏覽器提示您的連線不是私密連線Chrome瀏覽器
- 《四 資料庫連線池原始碼》手寫資料庫連線池資料庫原始碼