2023年4月11日 星期二

使用 Entity Framework 卻發生了連線池區 Connection Pool 已經超過最大大小的限制,導致系統噴出例外異常

使用 Entity Framework 卻發生了連線池區 Connection Pool 已經超過最大大小的限制,導致系統噴出例外異常

這是一個滿有趣的問題,也許是我寫的專案並沒有很大量的人在使用,所以,我幾乎沒有機會遇到如題目上的問題,也就是說,當取得一個 DbContext 物件的時候,當需要進行資料庫存取的時候,所使用 資料庫連線 Database Connection 資訊,其實是透過一個 連線集區 Connection Pool 來管理,因此,將會透過此集區來取的一個資料庫連線物件來使用,透過此資料庫連線物件,便可以對遠端資料庫系統進行資料存取操作,一旦對資料庫的存取完成之後,將會把這個資料庫連線物件歸還給連線集區內,如此可以達到重複使用的目的。

在一個偶然機會,聽到同事遇到一個狀況,那就是系統回報資料庫連線集區耗盡,因為時間逾期,最終結果將會造成了底下的例外異常的錯誤訊息;在還沒有寫這篇文章之前,我還真的是百思不得其解,基於好奇心又或者求知慾,讓我想要來做些實驗,究竟在什麼情況下,也就是做了甚麼樣的操作,會造成有這樣的 [The timeout period elapsed prior to obtaining a connection from the pool] 錯誤訊息發生,一旦有了這樣的知識之後,我在日後寫有關於資料庫存取的程式碼時候,將會積極、努力的來避免這樣的問題發生。

System.InvalidOperationException
  HResult=0x80131509
  Message=Timeout expired.  The timeout period elapsed prior to obtaining a connection from the pool.  This may have occurred because all pooled connections were in use and max pool size was reached.
  Source=Microsoft.Data.SqlClient
  StackTrace: 
   於 Microsoft.Data.ProviderBase.DbConnectionFactory.TryGetConnection(DbConnection owningConnection, TaskCompletionSource`1 retry, DbConnectionOptions userOptions, DbConnectionInternal oldConnection, DbConnectionInternal& connection)
   於 Microsoft.Data.ProviderBase.DbConnectionInternal.TryOpenConnectionInternal(DbConnection outerConnection, DbConnectionFactory connectionFactory, TaskCompletionSource`1 retry, DbConnectionOptions userOptions)
   於 Microsoft.Data.ProviderBase.DbConnectionClosed.TryOpenConnection(DbConnection outerConnection, DbConnectionFactory connectionFactory, TaskCompletionSource`1 retry, DbConnectionOptions userOptions)
   於 Microsoft.Data.SqlClient.SqlConnection.TryOpen(TaskCompletionSource`1 retry, SqlConnectionOverrides overrides)
   於 Microsoft.Data.SqlClient.SqlConnection.Open(SqlConnectionOverrides overrides)
   於 Microsoft.Data.SqlClient.SqlConnection.Open()
   於 EFCoreUsingDbContextPoolSize.Program.<>c__DisplayClass0_1.<<Main>b__0>d.MoveNext() 在 C:\Vulcan\Github\CSharp2023\EFCoreUsingDbContextPoolSize\EFCoreUsingDbContextPoolSize\Program.cs:行 83 中
   於 EFCoreUsingDbContextPoolSize.Program.<Main>d__0.MoveNext() 在 C:\Vulcan\Github\CSharp2023\EFCoreUsingDbContextPoolSize\EFCoreUsingDbContextPoolSize\Program.cs:行 112 中

建立測試用的專案

為了簡化測試用專案的複雜度,因此,在這裡將會建立一個 Console 主控台應用類型的專案。

  • 打開 Visual Studio 2022 IDE 應用程式
  • 從 [Visual Studio 2022] 對話窗中,點選右下方的 [建立新的專案] 按鈕
  • 在 [建立新專案] 對話窗右半部
    • 切換 [所有語言 (L)] 下拉選單控制項為 [C#]
    • 切換 [所有專案類型 (T)] 下拉選單控制項為 [主控台]
  • 在中間的專案範本清單中,找到並且點選 [主控台應用程式] 專案範本選項

    專案,用於建立可在 Windows、Linux 及 macOS 於 .NET 執行的命令列應用程式

  • 點選右下角的 [下一步] 按鈕
  • 在 [設定新的專案] 對話窗
  • 找到 [專案名稱] 欄位,輸入 EFCoreUsingDbContextPoolSize 作為專案名稱
  • 在剛剛輸入的 [專案名稱] 欄位下方,確認沒有勾選 [將解決方案與專案至於相同目錄中] 這個檢查盒控制項
  • 點選右下角的 [下一步] 按鈕
  • 現在將會看到 [其他資訊] 對話窗
  • 在 [架構] 欄位中,請選擇最新的開發框架,這裡選擇的 [架構] 是 : .NET 7.0 (標準字詞支援)
  • 在這個練習中,需要去勾選 [不要使用最上層陳述式(T)] 這個檢查盒控制項

    這裡的這個操作,可以由讀者自行決定是否要勾選這個檢查盒控制項

  • 請點選右下角的 [建立] 按鈕

稍微等候一下,這個主控台專案將會建立完成

安裝要用到的 NuGet 開發套件

因為開發此專案時會用到這些 NuGet 套件,請依照底下說明,將需要用到的 NuGet 套件安裝起來。

安裝 Microsoft.EntityFrameworkCore 套件

  • 滑鼠右擊 [方案總管] 視窗內的 [專案節點] 下方的 [相依性] 節點
  • 從彈出功能表清單中,點選 [管理 NuGet 套件] 這個功能選項清單
  • 此時,將會看到 [NuGet: EFCoreUsingDbContextPoolSize] 視窗
  • 切換此視窗的標籤頁次到名稱為 [瀏覽] 這個標籤頁次
  • 在左上方找到一個搜尋文字輸入盒,在此輸入 Microsoft.EntityFrameworkCore
  • 稍待一會,將會在下方看到這個套件被搜尋出來
  • 點選 [Microsoft.EntityFrameworkCore] 套件名稱
  • 在視窗右方,將會看到該套件詳細說明的內容,其中,右上方有的 [安裝] 按鈕
  • 點選這個 [安裝] 按鈕,將這個套件安裝到專案內

安裝 Microsoft.EntityFrameworkCore.SqlServer 套件

  • 滑鼠右擊 [方案總管] 視窗內的 [專案節點] 下方的 [相依性] 節點
  • 從彈出功能表清單中,點選 [管理 NuGet 套件] 這個功能選項清單
  • 此時,將會看到 [NuGet: EFCoreUsingDbContextPoolSize] 視窗
  • 切換此視窗的標籤頁次到名稱為 [瀏覽] 這個標籤頁次
  • 在左上方找到一個搜尋文字輸入盒,在此輸入 Microsoft.EntityFrameworkCore.SqlServer
  • 稍待一會,將會在下方看到這個套件被搜尋出來
  • 點選 [Microsoft.EntityFrameworkCore.SqlServer] 套件名稱
  • 在視窗右方,將會看到該套件詳細說明的內容,其中,右上方有的 [安裝] 按鈕
  • 點選這個 [安裝] 按鈕,將這個套件安裝到專案內

建立需要用到的資料項目模型

  • 滑鼠右擊 [方案總管] 視窗內的 [專案節點] 節點
  • 從彈出視窗中,依序點選 [加入] > [類別] 功能表選項
  • 此時,[新增項目] 對話窗將會出現
  • 在此對話窗的最下方 [名稱] 欄位內輸入 Student.cs 這個類別名稱

    請確認該對話窗中間的清單,已經確實選擇了 [類別] 這個項目

  • 點選此對話窗右下方的 [新增] 按鈕
  • 確認 [Student.cs] 這個視窗出現在 Visual Studio 2022 中
  • 使用底下 C# 程式碼替換掉剛剛產生的類別程式碼內容
namespace EFCoreUsingDbContextPoolSize;

public class Student
{
    public int StudentId { get; set; }
    public string Name { get; set; }
}
  • 滑鼠右擊 [方案總管] 視窗內的 [專案節點] 節點
  • 從彈出視窗中,依序點選 [加入] > [類別] 功能表選項
  • 此時,[新增項目] 對話窗將會出現
  • 在此對話窗的最下方 [名稱] 欄位內輸入 Course.cs 這個類別名稱

    請確認該對話窗中間的清單,已經確實選擇了 [類別] 這個項目

  • 點選此對話窗右下方的 [新增] 按鈕
  • 確認 [Course.cs] 這個視窗出現在 Visual Studio 2022 中
  • 使用底下 C# 程式碼替換掉剛剛產生的類別程式碼內容
namespace EFCoreUsingDbContextPoolSize;

public class Course
{
    public int CourseId { get; set; }
    public string CourseName { get; set; }
}
  • 滑鼠右擊 [方案總管] 視窗內的 [專案節點] 節點
  • 從彈出視窗中,依序點選 [加入] > [類別] 功能表選項
  • 此時,[新增項目] 對話窗將會出現
  • 在此對話窗的最下方 [名稱] 欄位內輸入 CouSchoolContextrse.cs 這個類別名稱

    請確認該對話窗中間的清單,已經確實選擇了 [類別] 這個項目

  • 點選此對話窗右下方的 [新增] 按鈕
  • 確認 [SchoolContext.cs] 這個視窗出現在 Visual Studio 2022 中
  • 使用底下 C# 程式碼替換掉剛剛產生的類別程式碼內容
using Microsoft.EntityFrameworkCore;

namespace EFCoreUsingDbContextPoolSize;

public class SchoolContext : DbContext
{
    public SchoolContext(DbContextOptions<SchoolContext> options)
    : base(options)
    {
    }

    public DbSet<Student> Students { get; set; }
    public DbSet<Course> Courses { get; set; }

    protected override void OnConfiguring(DbContextOptionsBuilder optionsBuilder)
    {
        //optionsBuilder.UseSqlServer("Data Source=(localdb)\\MSSQLLocalDB;Initial Catalog=TestPool;MultipleActiveResultSets=True");
    }
}

修正主程序的程式碼

  • 在此專案節點下,找到並且打開 [Program.cs] 這個檔案
  • 使用底下 C# 程式碼替換掉 [Program.cs] 檔案內所有程式碼內容
using Microsoft.EntityFrameworkCore;

namespace EFCoreUsingDbContextPoolSize;

/// <summary>
/// 模擬 DbContext 的 Connection Pool 超出使用量,造成例外異常的情況
/// </summary>
internal class Program
{
    static async Task Main(string[] args)
    {
        // 設定資料庫連線的最大連線數量
        int MaxPoolSize = 4;
        // 設定資料庫連線的最大等待時間
        int ConnectTimeout = 5;
        bool UsingEFCoreQuery = true;

        #region 準備進行資料庫連線或者建立資料庫
        // 連線字串
        string connectionString = @"Data Source=(localdb)\MSSQLLocalDB;Initial Catalog=TestPool;" +
            $"Max Pool Size={MaxPoolSize};Connect Timeout={ConnectTimeout};MultipleActiveResultSets=True";
        // 建立資料庫連線
        DbContextOptions<SchoolContext> options =
            new DbContextOptionsBuilder<SchoolContext>()
            .UseSqlServer(connectionString)
            .Options;

        using (var context = new SchoolContext(options))
        {
            // 檢查資料庫是否存在
            if (context.Database.CanConnect() == false)
            {
                // 建立資料庫
                await context.Database.EnsureCreatedAsync();

                context.Database.GetDbConnection();
                var connection = context.Database.GetDbConnection();
            }
        }
        #endregion

        #region 模擬建立多個 DbContext 來進行資料庫存取,故意造成 Connection Pool 被耗盡的情況

        List<Task> tasks = new List<Task>();

        for (int i = 0; i < 10; i++)
        {
            // 建立 10 個 DbContext 來進行資料庫存取
            int idx = i;
            Task task = Task.Run(async () =>
            {
                using (var context = new SchoolContext(options))
                {
                    if (UsingEFCoreQuery)
                    {
                        #region 使用 EF Core 的 SqlQuery 來進行查詢
                        Console.WriteLine($"第{idx}工作已經開始 " + DateTime.Now.ToString("mm:ss"));

                        var result = context.Database
                        .SqlQuery<string>($"SELECT GETDATE() AS D");
                        Console.WriteLine($"第{idx}工作準備休息 " + DateTime.Now.ToString("mm:ss"));
                        // 這裡休息多久,都不會有例外異常發生,因為,執行完 SQL 命令,連線就立即關閉了
                        Thread.Sleep(5000);
                        Console.WriteLine($"第{idx}工作正要結束 " + DateTime.Now.ToString("mm:ss"));

                        //var std = new Student()
                        //{
                        //    Name = "Bill"
                        //};
                        //context.Students.Add(std);
                        //context.SaveChanges();
                        #endregion
                    }
                    else
                    {
                        #region 使用 DbConnection 來存取資料庫,但是故意延遲關閉連線時間
                        using (var cn = context.Database.GetDbConnection())
                        {
                            Console.WriteLine($"第{idx}工作開啟連線 " + DateTime.Now.ToString("mm:ss"));
                            cn.Open();
                            // 關閉前休息3秒,並註解底下四行,同樣會造成例外異常
                            var cmd = cn.CreateCommand();
                            cmd.CommandText = "SELECT GETDATE() AS D";
                            var dr = cmd.ExecuteReader();
                            dr.Read();
                            Console.WriteLine($"第{idx}工作準備休息 " + DateTime.Now.ToString("mm:ss"));

                            // 這裡若將休息時間改為 3000 (3秒),將會造成底下錯誤
                            // -----------------------------------------------------
                            // System.InvalidOperationException: 'Timeout expired.
                            // The timeout period elapsed prior to obtaining a connection
                            // from the pool.  This may have occurred because
                            // all pooled connections were in use and max pool size was reached.'
                            Thread.Sleep(3000);
                            // 這裡換成 await Task.Delay 同樣無效,會造成例外異常
                            //await Task.Delay(3000);

                            Console.WriteLine($"第{idx}工作正要結束 " + DateTime.Now.ToString("mm:ss"));
                            cn.Close();
                        }
                        #endregion
                    }
                }
            });
            tasks.Add(task);
        }
        #endregion

        await Task.WhenAll(tasks);

        Console.WriteLine("Press any key for continuing...");
        Console.ReadKey();
    }
}

在這個進入點程式碼內,將會宣告三個物件: MaxPoolSize, ConnectTimeout, UsingEFCoreQuery

  • MaxPoolSize

    MaxPoolSize是連線集區的一個屬性,用於指定連線集區中可以容納的最大連接數。連線集區是一種性能優化技術,用於重用已建立的資料庫連接,以減少建立新連接所需的時間和資源。

    當應用程式需要與資料庫建立連接時,它會首先檢查連線集區中是否有可用的連接。如果有可用連接,應用程式將重用該連接,而不是建立新的連接。如果連線集區已滿(即達到 MaxPoolSize),則應用程式必須等待,直到連線集區中有連接可用。

    MaxPoolSize的預設值通常為 100,但可以根據應用程式的需求進行調整。將 MaxPoolSize 設置得太小可能會導致應用程式在高負載情況下性能下降,因為需要等待連接可用;將其設置得太大可能會導致資源浪費,因為可能無法充分利用所有連接。

  • ConnectTimeout

    設定資料庫連線的最大等待時間,這個物件值將會宣告在連線字串內,其中,連線字串中的Connect Timeout單位是秒(seconds)

    在C#中,如果在連線字串中將 ConnectTimeout 設置為5,則表示連接操作將在5秒後超時。如果在5秒內無法建立連接,操作將被終止,並且通常會拋出一個異常,例如SqlException或其他相應的異常,具體取決於您使用的資料庫類型。這樣可以防止程式無限期地等待無法建立的連接。

    若在連線字串內沒有指定 ConnectTimeout ,且使用 MSSQL Server 時,若在連線字串中沒有指定Connect Timeout,預設值為30秒。這表示如果在30秒內無法建立連接,操作將被終止。

  • UsingEFCoreQuery

    這個布林物件將會用於控制要採用 Entity Framework 的方式來存取資料庫紀錄,還是要使用 DbConnection 來存取資料庫,這裡會需要手動 Open 與 Close 這次的資料庫連線,用來模擬與分析不同的作法差別在哪裡。

接下來就是要宣告連線字串文字到 connectionString 物件內,這裡透過 MaxPoolSize & ConnectTimeout 兩個數值,宣告這個連線字串的連線物件執行特性。因此,便可以在這個程式內,調整這些物件值,反覆迭代執行程式,觀察看到不同結果,了解與分析結果。

緊接著將會透過剛剛建立好的連線字串,產生出一個 DbContextOptions 物件,有了這個物件,便可以接著來生成出 DbContext 物件,如此就可以來進行資料庫的紀錄存取動作囉。

string connectionString = @"Data Source=(localdb)\MSSQLLocalDB;Initial Catalog=TestPool;" +
            $"Max Pool Size={MaxPoolSize};Connect Timeout={ConnectTimeout};MultipleActiveResultSets=True";

由於這次使用的是 MSSQL Server 資料庫服務,因此,對於若在連線字串內沒有指定 [Max Pool Size] ,則會使用預設值 100;有更多關於這方面的資訊,另外,也可以使用 [Min Pool Size] 來指定此次資料庫存取的 Pool 內最少需要具備這麼多的連線數量存在,更多的資訊,可以參考 SQL Server Connection Pooling (ADO.NET)

對於在連線字串內用到的 [Connect Timeout] ,這裡指的是創建資料庫連接的超時時間,而不是通過該連接執行的命令的超時時間,因此,一旦想要取得一個資料庫連線動作需求,此時連線集區內沒有足夠的物件可供使用,而連線物件也尚未超過最大限制,此時,若無法在這裡指定時間內取得連線物件,則會造成例外異常拋出。更多資訊,可以參考 All SQL Server SqlConnection Properties

現在將會執行到底下程式碼

using (var context = new SchoolContext(options))
{
    // 檢查資料庫是否存在
    if (context.Database.CanConnect() == false)
    {
        // 建立資料庫
        await context.Database.EnsureCreatedAsync();
        context.Database.GetDbConnection();
        var connection = context.Database.GetDbConnection();
    }
}

在這裡透過剛剛建立的 DbContextOptions 類別物件, options 物件,建立出 SchoolContext 物件,透過這個物件首先檢查資料庫是否存在,若不存在的話,則會建立起一個新的資料庫;到此為止,這個程式一旦啟動之後,將會自動建立一個測試用的資料庫,並且這個資料庫內會擁有 [Course] & [Student] 這兩個 Table 資料表

找到 [#region 模擬建立多個 DbContext 來進行資料庫存取,故意造成 Connection Pool 被耗盡的情況] 這段程式碼區塊,這裡將會嘗試啟動 10 個工作(或者說是有 10 個工作背景執行緒),在每個執行緒內,將會建立一個 SchoolContext 物件,嘗試進行資料庫的存取動作

#region 可讓您指定在使用 Visual Studio 程式碼編輯器的大綱功能時,可以展開或摺疊的程式碼區塊。

首先,確認測試條件變數的物件值如下:

int MaxPoolSize = 4;
int ConnectTimeout = 5;
bool UsingEFCoreQuery = true;

這裡表示連線集區內最多僅會有 4 個連線物件,若這些連線物件被取出之後,沒有立即歸還,第五個需求者想要一個連線物件,不過超過了 5 秒之後都沒有取得連線物件(因為前面四個連線物件都遲遲不歸還),此時,應用程式將會拋出例外異常;最後一個變數則是宣告要使用 Entity Framework 的方式來進行資料庫存取。

底下為這樣要執行的程式碼內容

using (var context = new SchoolContext(options))
{
    if (UsingEFCoreQuery)
    {
        #region 使用 EF CoreSqlQuery 來進行查詢
        Console.WriteLine($"第{idx}工作已經開始 " + DateTime.Now.ToString("mm:ss"));
        var result = context.Database
        .SqlQuery<string>($"SELECT GETDATE() AS D");
        Console.WriteLine($"第{idx}工作準備休息 " + DateTime.Now.ToString("mm:ss"));
        // 這裡休息多久,都不會有例外異常發生,因為,執行完 SQL 命令,連線就立即關閉了
        Thread.Sleep(5000);
        Console.WriteLine($"第{idx}工作正要結束 " + DateTime.Now.ToString("mm:ss"));
        //var std = new Student()
        //{
        //    Name = "Bill"
        //};
        //context.Students.Add(std);
        //context.SaveChanges();
        #endregion
    }
    else
    {
      ...
    }
}

找到 [#region 使用 EF Core 的 SqlQuery 來進行查詢] 區段程式碼,可以看到這裡將會透過 context.Database.SqlQuery 送出 SELECT GETDATE() AS D 指定到資料庫內,不過,在送出該命令之後,會使用 Thread.Sleep(5000) 敘述來強制使得當前工作執行緒休息五秒,看看這樣是否會造成連線物件被霸占五秒,導致第六個執行緒因為遲遲無法取得連線物件,而導致系統崩潰。

現在,來實際執行這個測試程式,看看執行結果

第5工作已經開始 49:46
第1工作已經開始 49:46
第6工作已經開始 49:46
第2工作已經開始 49:46
第4工作已經開始 49:46
第7工作已經開始 49:46
第0工作已經開始 49:46
第3工作已經開始 49:46
第5工作準備休息 49:46
第4工作準備休息 49:46
第0工作準備休息 49:46
第6工作準備休息 49:46
第1工作準備休息 49:46
第7工作準備休息 49:46
第2工作準備休息 49:46
第3工作準備休息 49:46
第8工作已經開始 49:47
第8工作準備休息 49:47
第9工作已經開始 49:48
第9工作準備休息 49:48
第4工作正要結束 49:51
第7工作正要結束 49:51
第3工作正要結束 49:51
第5工作正要結束 49:51
第0工作正要結束 49:51
第1工作正要結束 49:51
第6工作正要結束 49:51
第2工作正要結束 49:51
第8工作正要結束 49:52
第9工作正要結束 49:53

從這裡執行結果,可以看到同時啟動 10 個工作執行緒,並不會受到連線集區最大連線物件 5 個限制有所影響

接下來,需要修改測試條件變數的物件值如下:

bool UsingEFCoreQuery = false;

在這裡將會執行底下的程式碼

using (var context = new SchoolContext(options))
{
    if (UsingEFCoreQuery)
    {
      ...
    }
    else
    {
        #region 使用 DbConnection 來存取資料庫,但是故意延遲關閉連線時間
        using (var cn = context.Database.GetDbConnection())
        {
            Console.WriteLine($"第{idx}工作開啟連線 " + DateTime.Now.ToString("mm:ss"));
            cn.Open();
            // 關閉前休息3秒,並註解底下四行,同樣會造成例外異常
            var cmd = cn.CreateCommand();
            cmd.CommandText = "SELECT GETDATE() AS D";
            var dr = cmd.ExecuteReader();
            dr.Read();
            Console.WriteLine($"第{idx}工作準備休息 " + DateTime.Now.ToString("mm:ss"));
            // 這裡若將休息時間改為 3000 (3秒),將會造成底下錯誤
            // -----------------------------------------------------
            // System.InvalidOperationException: 'Timeout expired.
            // The timeout period elapsed prior to obtaining a connection
            // from the pool.  This may have occurred because
            // all pooled connections were in use and max pool size was reached.'
            Thread.Sleep(3000);
            // 這裡換成 await Task.Delay 同樣無效,會造成例外異常
            //await Task.Delay(3000);
            Console.WriteLine($"第{idx}工作正要結束 " + DateTime.Now.ToString("mm:ss"));
            cn.Close();
        }
        #endregion
    }
}

首先透過 context.Database.GetDbConnection() 方法,取得 DbConnection 類別 物件。

接著,使用 Open() 方法來開啟資料庫連接,然後要使用 [ExecuteReader] 來執行 SELECT GETDATE() AS D SQL 查詢命令,經過執行了 [Read()] 方法之後,卻不立即執行 [Close()] 關閉連線方法,而是故意休息五秒鐘,這裡是要模擬也許執行了一個複雜的 SQL 命令,需要花費許多時間才能夠得到結果,又或者取得 SQL 命令執行結果後,但是,直接去處理其他事情,造成超過五秒以上的時間,最後才去呼叫關閉連線方法。

不論是哪個做法,都將造成從連線集區取得的連線物件沒有用完立即歸還,導致連線集區內沒有可用的連線物件,進而使得其他工作執行緒要進行資料庫存取動作時候,無法擁有一個連線物件來呼叫遠端資料庫 SQL 命令,最後因為超過 5 秒的逾時限制,使得該應用程式噴出例外異常錯誤,最終導致該應用程式執行失敗,無法繼續往下執行。

現在,實際來執行這個應用程式

底下是執行輸出內容

第1工作開啟連線 07:46
第2工作開啟連線 07:46
第6工作開啟連線 07:46
第7工作開啟連線 07:46
第5工作開啟連線 07:46
第3工作開啟連線 07:46
第0工作開啟連線 07:46
第4工作開啟連線 07:46
第2工作準備休息 07:46
第5工作準備休息 07:46
第1工作準備休息 07:46
第6工作準備休息 07:46
第8工作開啟連線 07:47
第9工作開啟連線 07:48
第2工作正要結束 07:49
第6工作正要結束 07:49
第1工作正要結束 07:49
第5工作正要結束 07:49
第3工作準備休息 07:49
第4工作準備休息 07:49
第7工作準備休息 07:49
第0工作準備休息 07:49
第4工作正要結束 07:52
第3工作正要結束 07:52
第7工作正要結束 07:52
第0工作正要結束 07:52
第9工作準備休息 07:52
第9工作正要結束 07:55

可是,這個時候,這個應用程式卻沒有執行完成,而得到了

System.InvalidOperationException: 'Timeout expired. The timeout period elapsed prior to obtaining a connection from the pool. This may have occurred because all pooled connections were in use and max pool size was reached.'

這樣的例外異常訊息,而 Visual Studio 2022 也停留在底下的畫面