iT邦幫忙

3

[C#] SQL 資料庫 Connection Pool 連線池觀念釐清

同步發表至 Blog:[C#] SQL 資料庫 Connection Pool 連線池觀念釐清

程式對資料庫存取是非常頻繁的操作,平常只會使用而沒有去深入了解底層的實作方式,這次花了點時間真正搞懂了 Connection Pool 的機制。

正常開關

static void Main(string[] args)
{
	// 設定相關資料庫連線參數
	string connection = @"Data Source=(LocalDb)\MSSQLLocalDB;Integrated Security=SSPI;AttachDBFilename=D:\Workspace\[03]Test\localdbtest\LocalDbTest\LocalDb.mdf;";

	Enumerable.Range(1, 10).Select(x => Task.Run(() =>
	{
		Stopwatch sw = new Stopwatch();
		sw.Start();
		
		SqlConnection sqlConnection = new SqlConnection();
		sqlConnection.ConnectionString = connection;

		// 開啟連線
		sqlConnection.Open();

		sqlConnection.Close();
		sqlConnection.Dispose();
		sqlConnection = null;

		sw.Stop();
		Console.WriteLine($"連線 {x} 共耗時 {sw.ElapsedMilliseconds} 毫秒");
	})).ToList();

	Console.ReadKey();
}

只開不關

static void Main(string[] args)
{
	// 設定相關資料庫連線參數
	string connection = @"Data Source=(LocalDb)\MSSQLLocalDB;Integrated Security=SSPI;AttachDBFilename=D:\Workspace\[03]Test\localdbtest\LocalDbTest\LocalDb.mdf;";

	Enumerable.Range(1, 10).Select(x => Task.Run(() =>
	{
		Stopwatch sw = new Stopwatch();
		sw.Start();

		SqlConnection sqlConnection = new SqlConnection();
		sqlConnection.ConnectionString = connection;

		// 開啟連線
		sqlConnection.Open();

		sw.Stop();
		Console.WriteLine($"連線 {x} 共耗時 {sw.ElapsedMilliseconds} 毫秒");
	})).ToList();

	Console.ReadKey();
}

從上面的兩個例子發現前四個消耗的秒數特別多,可能是因為初始化連線時所耗的效能導致此結果。第一個例子後面六個連線會發現都在 0-1 ms 之間;第二個例子後面六個連線都耗費了 4-16 ms 不等,由此可判斷程式自動幫我使用了連線池來增進效能。

如果連線字串加上 Max Pool Size=5 (預設為 100) 則可以限制最多開啟五條連線,高負載的應用程式可以借由調整此參數增進效能。

限制連線池大小

static void Main(string[] args)
{
	// 設定相關資料庫連線參數
	string connection = @"Data Source=(LocalDb)\MSSQLLocalDB;Integrated Security=SSPI;AttachDBFilename=D:\Workspace\[03]Test\localdbtest\LocalDbTest\LocalDb.mdf;Max Pool Size=5";

	Enumerable.Range(1, 10).Select(x => Task.Run(() =>
	{
		Stopwatch sw = new Stopwatch();
		sw.Start();

		SqlConnection sqlConnection = new SqlConnection();
		sqlConnection.ConnectionString = connection;

		// 開啟連線
		sqlConnection.Open();

		sw.Stop();
		Console.WriteLine($"連線 {x} 共耗時 {sw.ElapsedMilliseconds} 毫秒");
	})).ToList();

	Console.ReadKey();
}

關閉連線池(每次連線都建立新的連線)

static void Main(string[] args)
{
	// 設定相關資料庫連線參數
	string connection = @"Data Source=(LocalDb)\MSSQLLocalDB;Integrated Security=SSPI;AttachDBFilename=D:\Workspace\[03]Test\localdbtest\LocalDbTest\LocalDb.mdf;Max Pool Size=5;Pooling=False;";

	Enumerable.Range(1, 10).Select(x => Task.Run(() =>
	{
		Stopwatch sw = new Stopwatch();
		sw.Start();

		SqlConnection sqlConnection = new SqlConnection();
		sqlConnection.ConnectionString = connection;

		// 開啟連線
		sqlConnection.Open();

		sw.Stop();
		Console.WriteLine($"連線 {x} 共耗時 {sw.ElapsedMilliseconds} 毫秒");
	})).ToList();

	Console.ReadKey();
}

上面兩個例子可以看出限制連線數的情況下只開啟不關閉,會發生連線無法重用導致達到上限無法建立新連線,可能有人會說那關閉連線池不就得了?

但不要忘了一台電腦擁有的端口是有限的(65535),若超出上限你的程式還是會大爆炸,所以切記在撰寫資料庫程式的時候一定要保持隨手關連線的好習慣。

如果想要清除所有連線的話可以使用 SqlConnection.ClearAllPools() 強制清除連線池中未使用的連線

static void Main(string[] args)
{
	// 設定相關資料庫連線參數
	string connection = @"Data Source=(LocalDb)\MSSQLLocalDB;Integrated Security=SSPI;AttachDBFilename=D:\Workspace\[03]Test\localdbtest\LocalDbTest\LocalDb.mdf;Max Pool Size=5;";

	Enumerable.Range(1, 10).Select(x => Task.Run(() =>
	{
		Stopwatch sw = new Stopwatch();
		sw.Start();

		SqlConnection sqlConnection = new SqlConnection();
		sqlConnection.ConnectionString = connection;

		// 開啟連線
		sqlConnection.Open();

		sqlConnection.Close();
		sqlConnection.Dispose();
		sqlConnection = null;
		SqlConnection.ClearAllPools();

		sw.Stop();
		Console.WriteLine($"連線 {x} 共耗時 {sw.ElapsedMilliseconds} 毫秒");
	})).ToList();

	Console.ReadKey();
}

參考:

  1. SQL Server 連接共用 (ADO.NET)
  2. 正确理解 SqlConnection 的连接池机制
  3. 超时时间已到.错误及Max Pool Size设置

1 則留言

0
bojing8508
iT邦新手 5 級 ‧ 2020-09-23 13:02:51

請教攻城屍大大,WEB出現以下訊息與connection pool有關還是純粹網路不好?

錯誤訊息:

已超過連接逾時的設定。在取得集區連接之前超過逾時等待的時間,可能的原因為所有的共用連接已在使用中,並已達共用集區大小的最大值

當掉的點是,只要在web新增大概15筆資料,網頁就會出現上述訊息
進入效能檢測觀察 user connrections 超過100,網頁就無法運作
重整IIS WEB才能恢復正常。

直接登入VM操作也同樣情況。

附上https://ithelp.ithome.com.tw/upload/images/20200923/201139674xMEhnHYpn.png

我要留言

立即登入留言