因海關業務複雜,資料量極大,在做sql自訂查詢的時候需要限定返回結果集DataSet的資料量,微軟本身的DataAdpter的Fill方法根本無法滿足其需要,所以得自己想法實現,再自己實現了IDataReader轉換為DataSet的方法後,發現在調用DataReader.Close的時候老是報“逾時異常”。仔細翻閱了MSDN對Close方法的說明後,在備忘的地方發現了這麼一段話:
Close 方法填寫輸出參數的值、傳回值和 RecordsAffected,從而增加了關閉用於處理大型或複雜查詢的 SqlDataReader 所用的時間。如果傳回值和查詢影響的記錄的數量不重要,則可以在調用 Close 方法前調用關聯的 SqlCommand 對象的 Cancel 方法,從而減少關閉 SqlDataReader 所需的時間。
DataReader為了計算RecordsAffected值,在Close之前會繼續讀取剩下的記錄,直到讀完為止。原來執行Command的Cancel方法就可以解決這個問題。
下面是我的測試函數代碼(僅供參考):
| 代碼如下 |
複製代碼 |
public static void FillDataSet2(DataSet ds, DbCommand cmd, int pageSize, int pageIndex) { DataTable table = new DataTable(); using (IDataReader reader = cmd.ExecuteReader(CommandBehavior.CloseConnection)) { int fieldCount = reader.FieldCount; for (int i = 0; i < fieldCount; i++) { table.Columns.Add(reader.GetName(i), reader.GetFieldType(i)); } ds.Tables.Add(table); object[] values = new object[fieldCount]; int currentIndex = 0; int startIndex = pageSize * pageIndex; try { table.BeginLoadData(); while (reader.Read()) { if (startIndex > currentIndex++) continue; if (pageSize > 0 && (currentIndex – startIndex) > pageSize) break; reader.GetValues(values); table.LoadDataRow(values, true); } } finally { table.EndLoadData(); //注意調用Cancel()方法:讀取一半的時候調用SqlDataReader.Close()來關閉讀取.結果老是提示操作逾時。 //Msdn解釋: //Close 方法填寫輸出參數的值、傳回值和 RecordsAffected,從而增加了關閉用於處理大型或複雜查詢的 SqlDataReader 所用的時間。 //如果傳回值和查詢影響的記錄的數量不重要,則可以在調用 Close 方法前調用關聯的 SqlCommand 對象的 Cancel 方法,從而減少關閉 SqlDataReader 所需的時間。 cmd.Cancel(); reader.Close(); } } }
|
解決問題,效率也還可以。
註:之所以不用實體和高效率的分頁方法,是因為sql語句為業務人員自己寫的,無法確定有哪些欄位,語句複雜度也無法確定,所以採用此方法。