前言 由於項目需求,需要將Excel中的數據進過一定轉換導入僅Oracle資料庫中。考慮到當Excel數據量較大時,迴圈Insert語句效率太低,故採用批量插入的方法。在插入操作運行時,會造成系統短暫的“卡死”現象。為了讓用戶知道插入的狀態,需要製作一個進度條來顯示插入的進度。 批量插入 項目中運用 ...
前言
由於項目需求,需要將Excel中的數據進過一定轉換導入僅Oracle資料庫中。考慮到當Excel數據量較大時,迴圈Insert語句效率太低,故採用批量插入的方法。在插入操作運行時,會造成系統短暫的“卡死”現象。為了讓用戶知道插入的狀態,需要製作一個進度條來顯示插入的進度。
批量插入
項目中運用的是System.Data.OracleClient。首先將Excel數據通過轉換函數轉換為DataTable,其中的欄位和資料庫中相應表格的欄位完全對應。
public int Import2Oracle(ISheet sheet, string tablename)
{
DataTable dt = GetDataFromExcelByNPOI(sheet); //經過轉換後獲得DataTable
OracleCommand cmd = conn.CreateCommand();// conn為資料庫連接對象
OracleDataAdapter da = new OracleDataAdapter(cmd);
OracleCommandBuilder ocb = new OracleCommandBuilder(da);
string SelectSQL = "select * from "+tablename+ " where ROWNUM=0";
da.SelectCommand.CommandText = SelectSQL;
da.InsertCommand = ocb.GetInsertCommand();
da.Update(dt);
return 1; //返回正常
}
上述代碼沒有列出連接Oracle資料庫相關代碼,該段代碼將DataTable批量插入資料庫中。為了利用進度條控制項實現顯示插入的進度,需要在每次成功插入一條數據後更新進度條。
進度條實現
為了實現實時更新插入進度,需要用到OracleDataAdapter類下RowUpdated事件。官方文檔中有註釋,當使用Update方法時,在每一條記錄更新時會發生兩個事件,即OnRowUpdating和OnRowUpdated。利用OnRowUpdated這個事件即可記錄已插入多少條數據,顯示插入進度。
int Sum = 0; //總記錄數
int rowNum = 0; //插入記錄數
public int Import2Oracle(ISheet sheet)
{
Sum = sheet.LastRowNum - 1; //獲取總記錄數
DataTable dt = GetDataFromExcelByNPOI(sheet); //經過轉換後獲得DataTable,利用到了NPOI
OracleCommand cmd = conn.CreateCommand();// conn為資料庫連接對象
OracleDataAdapter da = new OracleDataAdapter(cmd);
OracleCommandBuilder ocb = new OracleCommandBuilder(da);
string SelectSQL = "select * from GZGDZL."+tablename+ " where ROWNUM=0";
da.SelectCommand.CommandText = SelectSQL;
da.InsertCommand = ocb.GetInsertCommand();
// update, this operation fires two events
// (RowUpdating/RowUpdated) per changed row
//這裡只用到RowUpdated事件
da.RowUpdated += new OracleRowUpdatedEventHandler(OnRowUpdated);
da.Update(dt);
return 1; //返回正常
}
//OnRowUpdated事件
private void OnRowUpdated(object sender, OracleRowUpdatedEventArgs e)
{
//刷新界面進度條
rowNum = rowNum + 1;
reportValue = (int)(rowNum /m_sheet.LastRowNum * 100);
this.progressBar1.Value = percent; //progressBar控制項已經設置最大值為100,最小值為0
}
經過改進後以上代碼即可讓用戶實時看到數據插入進度,雖然在一定程度上會影響插入的效率,但換來了用戶對插入狀態的瞭解,而且這種影響在數據量不是特別巨大的情況下幾乎可以忽略。