ADO.NET的记忆碎片(二)
Command類在執行SQL的時候,可能會遇到一次執行多條SQL的情況,就像這樣:
string strConn ="...";//連接字符串
string strSql = "update tableName1 set...;"+"update tableName2 set...;";
SqlConnection cn = new SqlConnection(strConn);
cn.Open();
SqlCommand cmd = new SqlCommand(strSql,cn);
int intTotalRowsAffected = cmd.ExecuteNonQuery();//返回的是多條SQl語句影響的總行數
可以看到返回的是影響數據庫表的總行數,這并不是我們想要的結果,解決的方法是在每次執行一條SQL語句完成的時候就能返回一個數據給我,這才是我想的。好在ADO.NET2.0提供了我們這樣的獲取機會,在Command中公開了一個StatementCompleted事件,這個事件是在每一條SQL語句執行完成后促發的,并且事件的主要參數中提供了RecordCount屬性,就是表示這次SQL語句的執行對數據庫表所產生的行數的影響。實現該功能如下:
string strSql = "update tableName1 set...;"+"update tableName2 set...;";
SqlConnection cn = new SqlConnection(strConn);
cn.Open();
SqlCommand cmd = new SqlCommand(strSql,cn);
cmd.StatementCompleted += new StatementCompletedEventHandler(HandleStatementCompleted);
int intTotalRowsAffected = cmd.ExecuteNonQuery();//返回的是多條SQl語句影響的總行數
cn.Close();
...
static void HandleStatementCompleted(object sender,StatementCompletedEventArgs e)
{
??? Console.WriteLine(e.RecordCount);
}
DataReader類
前面其實已經多次使用了DataReader,在此做一回顧:
當一個Command對象執行ExecuteReader()方法返回值就是一個DataReader類型的實例,保存著查詢的結果集。代碼如下:
SqlDataReader rdr = cmd.ExecuteReader();
對查詢的結果集遍歷:
while(rdr.Read())
{
??? Console.WriteLine("{0}--{1}",rdr[0],rdr["CustomerName"]);? ?
}
rdr.Close();
好了,下面對DataReader類進行進一步的研究,很多時候我們得到結果集,但是想看看里面的字段的數目、字段名稱、字段的Net類型、字段的數據庫數據類型等,這一些架構信息可以幫助我們更好的擴展應用程序,ADO.NET也幫助我們實現了方法我們可以自由的調用:
for(int intField=0;intField<rdr.FieldCount;intField++)//FieldCount是讀取字段數量的屬性
{
??? Console.WriteLine(intField);
??? Console.WriteLine(rdr.GetName(intField));//獲取字段名稱,參數是該字段的索引
??? Console.WriteLine(rdr.GetFieldType(intField).Name);//獲取net數據類型,參數是該字段的索引
??? Console.WriteLine(rdr.GetDataTypeName(intField));//獲取數據庫數據類型,參數是該字段的索引
??? Console.WriteLine();
}
rdr.Close();
知道了字段名稱怎么確定字段的索引,可以使用GetOrdinal()方法:
int fieldInt = rdr.GetOrdinal("CustomerName");//返回CustomerName這個字段的索引
用基于序號的查閱字段內容比基于字段名稱查閱性能好,所有就是因為這個原因有時候GetOrdinal()是非常受歡迎的。
原來是:
while(rdr.Read())
{
??? Console.WriteLine("{0}--{1}",rdr[0],rdr["CustomerName"]);? ?
}
rdr.Close();
可以改成:
int fieldInt = rdr.GetOrdinal("CustomerName");
while(rdr.Read())
{
??? Console.WriteLine("{0}",rdr[fieldInt]);? ?
}
rdr.Close();
這樣性能就可以得到提升,其實這段代碼的性能還可以進一步的提升。
強類型的getter
其實目前我們使用的rdr[0]基于索引的取值,其實是返回一個object類型的數值,但是我們本來的字段的類型是多樣的,有:int,string,datetime等,這樣其實里面會有一個類型轉換的問題,術語是會“裝箱”和“拆箱”,這是很影響性能的,解決的辦法就是使用強類型的getter,里面定義了許多net數據類型的公開Get(),如:GetString(),GetInt32()? ,GetDateTime().代碼可以改變如下:
int fieldInt = rdr.GetOrdinal("CustomerName");
while(rdr.Read())
{
??? Console.WriteLine("{0}",rdr.GetString(fieldInt));? ?
}
rdr.Close();
在開發過程中,如果不能確定字段的類型,請嘗試調用GetFieldType().
處理來自查詢的多個結果集
很多時候我們一次查詢會返回多個select結果,但是用DataRead實例中的Read()方法只能訪問第一個結果集,要訪問其他的結果集要調用NextResult()方法。代碼:
{
??? while(rdr.Read())
?? {
??????? Console.WriteLine("{0}",rdr.GetString(fieldInt));? ?
?? }
}while(rdr.NextResult());
?
轉載于:https://www.cnblogs.com/lmfeng/archive/2011/12/01/2270982.html
《新程序員》:云原生和全面數字化實踐50位技術專家共同創作,文字、視頻、音頻交互閱讀總結
以上是生活随笔為你收集整理的ADO.NET的记忆碎片(二)的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Linux 命令 - od
- 下一篇: [wiki]红黑树