问题:
为什么以下代码会泄漏连接?
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 服务器活动监视器,我会看到会话 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.关闭();
get被执行,所以问题不在于
if(sqlcon.State != System.Data.ConnectionState.Closed)
连接池。 别担心。
这是正常行为。
http://msdn.microsoft.com/en-us/library/8xx3tyca(v=vs.100).aspx
ADO.Net 池连接,以便可以重用它们,因为它们的创建成本相对较高。
连接到数据库服务器通常包括几个耗时的步骤。必须建立套接字或命名管道等物理通道,必须与服务器进行初始握手,必须分析连接字符串信息,必须由服务器对连接进行身份验证,必须运行检查以在当前事务中登记,等等。
实际上,大多数应用程序仅使用一种或几种不同的配置进行连接。这意味着在应用程序执行期间,将重复打开和关闭许多相同的连接。为了最大限度地降低打开连接的成本,ADO.NET 使用了一种称为连接池的优化技术。
http://msdn.microsoft.com/en-us/library/8xx3tyca(v=vs.100).aspx
此外,无需显式调用 。关闭()。 你的使用块将调用IDisposable.Dispose(),这将正确关闭连接。