執行長時間任務查詢數據庫,那么可能會出現一個問題:在執行長時間任務的過程中,如果點擊了進度條中的“取消”按鈕,此時可以停止程序執行查詢數據庫的任務,但是如果再次執行該任務時,可能會報錯:“連接未關閉,連接的當前狀態為打開”。下面以一個簡單的示例來看看出現上面所述問題的原因所在。首先,用access創建一個數據庫文件database.mdb,并保存在C#項目的bin\debug文件夾中。創建一個數據表TestData,在數據表里設置一個自增字段、一個姓名字段,一個年齡字段。仍然使用上一篇文章中所創建窗體和類,只把DataOperate.cs這個類修改一下,改成以下的代碼,用于模擬耗時的數據庫讀取任務:using System;
using System.ComponentModel;
using System.Data.OleDb;
using System.Data;
using System.IO;
using System.Threading;
namespace Test
{
internal class DataOperate
{
private static String StrConn;
private static OleDbConnection conn;
private static OleDbCommand sc;
public void operating(BackgroundWorker worker)
{
ConnectDatabase();
for (int i = 0; i < 10; i++)
{
Thread.Sleep(1000); //暫停1秒,模擬耗時的數據庫操作
operate("insert into TestData(姓名,年齡) values('全棧開發的碼農',3)");
worker.ReportProgress(i*100/10);
}
}
private void ConnectDatabase()
{
//連接數據庫
StrConn = "Provider=Microsoft.Jet.OLEDB.4.0;Data Source=" + Directory.GetCurrentDirectory() + @"\database.mdb; " + "Persist Security Info=True;Jet OLEDB:Database Password=123456";
try
{
conn = new OleDbConnection(StrConn);
}
catch (System.Data.SqlClient.SqlException ex)
{
throw new Exception(ex.Message);
}
}
private void operate(String strsql)//插入、刪除、更新等數據庫操作
{
conn.Open();
sc = new OleDbCommand();
sc.CommandText = strsql;
sc.Connection = conn;
Thread.Sleep(1000); //暫停1秒,模擬耗時的數據庫操作
sc.ExecuteNonQuery();
conn.Close();
}
public void close() //關閉數據庫連接
{
if (conn.State == ConnectionState.Open)
{
conn.Close();
conn.Dispose();
}
}
}
}
此時,點擊主窗體中的“開始”按鈕,開始執行任務,往數據庫里的TestData數據表插入記錄,程序運行正常。執行到一半的時候,點擊進度條窗體中的“取消”按鈕,此時任務被取消。此時,如果再次點擊主窗體中的“開始”按鈕,出現“連接未關閉”的報錯。經分析,問題出現在OleDbConnection對象conn上,在DataOperate.cs類中,把OleDbConnection對象conn定義成static靜態對象,這對于一般的數據庫操作是沒有問題的,因為每一個數據庫執行方法中都有conn.open()和conn.close(),數據連接可以正常打開和正常關閉,而當使用了Backgroundworker對象進行進度報告時,中途突然中斷數據庫查詢操作,那么會導致conn.close()無法正常執行,使得連接依然處于打開狀態,下一次嘗試再次連接時就出現了“連接未關閉”的報錯。解決問題的最簡單粗暴的方法就是把static刪除,使conn成為一個普通的非靜態變量即可。這是因為static對象在程序運行的全過程中只能有一個實例,第一次執行數據庫操作時,它已經實例化,再次執行數據庫操作時,也只能使用它,不能再次實例化,而它沒有被正常關閉連接,所以再次嘗試連接數據庫時會出現“連接未關閉”的情況,而如果采用了非靜態變量,那么就相當于給conn對象重新實例化,用一個全新的完全不同于此前的實例來操作數據庫,這樣就不會報錯了。
閱讀原文:原文鏈接
該文章在 2025/1/21 9:36:19 編輯過