執(zhí)行長(zhǎng)時(shí)間任務(wù)查詢數(shù)據(jù)庫(kù),那么可能會(huì)出現(xiàn)一個(gè)問(wèn)題:在執(zhí)行長(zhǎng)時(shí)間任務(wù)的過(guò)程中,如果點(diǎn)擊了進(jìn)度條中的“取消”按鈕,此時(shí)可以停止程序執(zhí)行查詢數(shù)據(jù)庫(kù)的任務(wù),但是如果再次執(zhí)行該任務(wù)時(shí),可能會(huì)報(bào)錯(cuò):“連接未關(guān)閉,連接的當(dāng)前狀態(tài)為打開(kāi)”。下面以一個(gè)簡(jiǎn)單的示例來(lái)看看出現(xiàn)上面所述問(wèn)題的原因所在。首先,用access創(chuàng)建一個(gè)數(shù)據(jù)庫(kù)文件database.mdb,并保存在C#項(xiàng)目的bin\debug文件夾中。創(chuàng)建一個(gè)數(shù)據(jù)表TestData,在數(shù)據(jù)表里設(shè)置一個(gè)自增字段、一個(gè)姓名字段,一個(gè)年齡字段。仍然使用上一篇文章中所創(chuàng)建窗體和類,只把DataOperate.cs這個(gè)類修改一下,改成以下的代碼,用于模擬耗時(shí)的數(shù)據(jù)庫(kù)讀取任務(wù):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秒,模擬耗時(shí)的數(shù)據(jù)庫(kù)操作
operate("insert into TestData(姓名,年齡) values('全棧開(kāi)發(fā)的碼農(nóng)',3)");
worker.ReportProgress(i*100/10);
}
}
private void ConnectDatabase()
{
//連接數(shù)據(jù)庫(kù)
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)//插入、刪除、更新等數(shù)據(jù)庫(kù)操作
{
conn.Open();
sc = new OleDbCommand();
sc.CommandText = strsql;
sc.Connection = conn;
Thread.Sleep(1000); //暫停1秒,模擬耗時(shí)的數(shù)據(jù)庫(kù)操作
sc.ExecuteNonQuery();
conn.Close();
}
public void close() //關(guān)閉數(shù)據(jù)庫(kù)連接
{
if (conn.State == ConnectionState.Open)
{
conn.Close();
conn.Dispose();
}
}
}
}
此時(shí),點(diǎn)擊主窗體中的“開(kāi)始”按鈕,開(kāi)始執(zhí)行任務(wù),往數(shù)據(jù)庫(kù)里的TestData數(shù)據(jù)表插入記錄,程序運(yùn)行正常。執(zhí)行到一半的時(shí)候,點(diǎn)擊進(jìn)度條窗體中的“取消”按鈕,此時(shí)任務(wù)被取消。此時(shí),如果再次點(diǎn)擊主窗體中的“開(kāi)始”按鈕,出現(xiàn)“連接未關(guān)閉”的報(bào)錯(cuò)。經(jīng)分析,問(wèn)題出現(xiàn)在OleDbConnection對(duì)象conn上,在DataOperate.cs類中,把OleDbConnection對(duì)象conn定義成static靜態(tài)對(duì)象,這對(duì)于一般的數(shù)據(jù)庫(kù)操作是沒(méi)有問(wèn)題的,因?yàn)槊恳粋€(gè)數(shù)據(jù)庫(kù)執(zhí)行方法中都有conn.open()和conn.close(),數(shù)據(jù)連接可以正常打開(kāi)和正常關(guān)閉,而當(dāng)使用了Backgroundworker對(duì)象進(jìn)行進(jìn)度報(bào)告時(shí),中途突然中斷數(shù)據(jù)庫(kù)查詢操作,那么會(huì)導(dǎo)致conn.close()無(wú)法正常執(zhí)行,使得連接依然處于打開(kāi)狀態(tài),下一次嘗試再次連接時(shí)就出現(xiàn)了“連接未關(guān)閉”的報(bào)錯(cuò)。解決問(wèn)題的最簡(jiǎn)單粗暴的方法就是把static刪除,使conn成為一個(gè)普通的非靜態(tài)變量即可。這是因?yàn)閟tatic對(duì)象在程序運(yùn)行的全過(guò)程中只能有一個(gè)實(shí)例,第一次執(zhí)行數(shù)據(jù)庫(kù)操作時(shí),它已經(jīng)實(shí)例化,再次執(zhí)行數(shù)據(jù)庫(kù)操作時(shí),也只能使用它,不能再次實(shí)例化,而它沒(méi)有被正常關(guān)閉連接,所以再次嘗試連接數(shù)據(jù)庫(kù)時(shí)會(huì)出現(xiàn)“連接未關(guān)閉”的情況,而如果采用了非靜態(tài)變量,那么就相當(dāng)于給conn對(duì)象重新實(shí)例化,用一個(gè)全新的完全不同于此前的實(shí)例來(lái)操作數(shù)據(jù)庫(kù),這樣就不會(huì)報(bào)錯(cuò)了。
閱讀原文:原文鏈接
該文章在 2025/1/21 9:36:19 編輯過(guò)