簡體   English   中英

為什么我的代碼泄漏連接?

[英]Why is my code leaking connections?

題:

為什么以下代碼會泄漏連接?

    public System.Data.DataTable GetDataTable()
    {
        System.Data.DataTable dt = new System.Data.DataTable();
        string strConnectionString = "Data Source=localhost;Initial Catalog=MyDb;User Id=SomeOne;Password=TopSecret;Persist Security Info=False;MultipleActiveResultSets=False;Packet Size=4096;";
        System.Data.SqlClient.SqlConnectionStringBuilder csb = new System.Data.SqlClient.SqlConnectionStringBuilder(strConnectionString);
        csb.IntegratedSecurity = true;


        string strSQL = "SELECT * FROM T_Benutzergruppen";

        using (System.Data.SqlClient.SqlConnection sqlcon = new System.Data.SqlClient.SqlConnection(csb.ConnectionString))
        {
            using (System.Data.SqlClient.SqlCommand cmd = new System.Data.SqlClient.SqlCommand(strSQL, sqlcon))
            {
                if (sqlcon.State != System.Data.ConnectionState.Open)
                {
                    sqlcon.Open();
                }

                // First attempt
                //System.Data.SqlClient.SqlDataAdapter sqlda = new System.Data.SqlClient.SqlDataAdapter("SELECT * FROM T_Benutzer", sqlcon);
                //sqlda.Fill(dt);

                cmd.ExecuteNonQuery();
            }

            if(sqlcon.State != System.Data.ConnectionState.Closed)
                sqlcon.Close();
        }
        //sqlcon.ConnectionString = csb.ConnectionString;

        // Second attempt
        //System.Data.SqlClient.SqlDataAdapter sqlda = new System.Data.SqlClient.SqlDataAdapter("SELECT * FROM T_Benutzer", csb.ConnectionString);
        //sqlda.Fill(dt);


        return dt;
    }

如果我進入SQL-Server活動監視器,則看到會話68

SELECT * FROM T_Benutzergruppen

附加問題:

如果有問題:

如果我注釋掉ConnectionStringBuilder以外的所有內容,並且僅在此函數中執行以下代碼,為什么它也會泄漏連接?

// Second attempt
System.Data.SqlClient.SqlDataAdapter sqlda = new System.Data.SqlClient.SqlDataAdapter("SELECT * FROM T_Benutzer", csb.ConnectionString);
sqlda.Fill(dt);

注意:
executenonquery毫無意義,僅用於測試目的。

如果我讓它在調試器中運行,則會看到sqlcon.Close();。

得到執行,所以問題不在於

if(sqlcon.State != System.Data.ConnectionState.Closed)

連接池。 不用擔心

這是正常現象。

http://msdn.microsoft.com/zh-CN/library/8xx3tyca(v=vs.100).aspx

ADO.Net合並連接,以便可以重新使用它們,因為它們創建起來比較昂貴。

連接到數據庫服務器通常包括幾個耗時的步驟。 必須建立一個物理通道(例如套接字或命名管道),必須與服務器進行初始握手,必須解析連接字符串信息,必須由服務器對連接進行身份驗證,必須運行檢查才能在當前列表中注冊交易等等。

實際上,大多數應用程序僅使用一種或幾種不同的配置進行連接。 這意味着在應用程序執行期間,許多相同的連接將被反復打開和關閉。 為了最大程度地減少打開連接的成本,ADO.NET使用了一種稱為連接池的優化技術。

http://msdn.microsoft.com/zh-CN/library/8xx3tyca(v=vs.100).aspx

同樣,也無需顯式調用.Close()。 您的using塊將調用IDisposable.Dispose() ,它將正確關閉連接。

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM