使用ADO.NET輕鬆操縱資料庫(一)

javaprogramers發表於2005-04-21

  ADO.NET提供了Connection來連線資料庫,同時也提供了Command物件來查詢資料庫。同Connection物件一樣,Command也有兩種:OleDbCommand和SqlCommand.其區別同Connection物件。

  要操縱資料庫,必須先使用Connection來連線到資料庫,再建立一個Command來查詢。有幾種建立方式,例:

SqlCommand cmd;

string strCon="server=localhost;database=Northwind;Trusted_Connection=Yes;";
string strqry="select * from Categories";
SqlConnection con=new SqlConnection(strCon);
con.Open();
¹cmd=con.CreateCommand(); //這裡使用用Connection物件的CreateCommand方法來建立一個Command物件。
cmd.CommandText=strqry;
//SqlDataReader reader=cmd.ExecuteReader();

cmd=new SqlCommand();?? //直接使用new 關鍵字來建立
cmd.CommandText=strqry;
?cmd.Connection=con;
?? //設定與資料庫的連線

 ³cmd=new SqlCommand(strqry,con); //直接在new的時候帶兩個引數來建立


 執行方式:

(主要有這麼幾種,cmd.ExecuteReader();cmd.ExecuteNonQuery();cmd.ExecuteScalar();cmd.ExecuteXmlReader();)

 1,ExecuteReader();返回一個SqlDataReader物件或OleDbDataReader物件,這個看你的程式的需要去  做。可以通過這個物件來檢查查詢結果,它提供了“游水”式的執行方式,即從結果中讀取一行之後,移動到另一行,則前一行就無法再用。有一點要注意的是執行之後,要等到手動去呼叫Read()方法之後,DataReader物件才會移動到結果集的第一行,同時此方法也返回一個Bool值,表明下一行是否可用,返回True則可用,返回False則到達結果集末尾。

使用DataReader可以提高執行效率,有兩種方式可以提高程式碼的效能:一種是基於序號的查詢,一個是使用適當的Get方法來查詢。因為查詢出來的結果一般都不會改變,除非再次改動查詢語句,因此可以通過定位列的位置來查詢記錄。用這種方法有一個問題,就是可能知道一列的名稱而不知道其所在的位置,這個問題的解決方案是通過呼叫DataReader 物件的GetOrdinal()方法,此方法接收一個列名並返回此列名所在的列號。例:

int id=reader.GetOrdinal("CategoryName");
while(reader.Read())
{
Response.Write(reader[id]);

reader.Close();

?至於第二種方式很直觀,例:

 while(reader.Read())
{
  ?Response.Write(reader.GetInt32(0).ToString()+" "+reader.GetString(1).ToString()+"
");
}

 DataReader的GetInt32()和GetString()通過接收一個列號來返回一個列的值,這兩種是最常用的,其中  還有很多其它的型別。

(注:DataReader物件在呼叫Close()方法即關閉與資料庫的連線,如果在沒有關閉之前又重新開啟第二個連線,則會產生一條異常資訊)

 2.,ExecuteNonQuery()?這個方法並不返回一個DataReader物件,而是返回一個int型別的值,即在執行之後在資料庫中所影響的行數。

 例:

 int affectrows=cmd.ExecuteNonQuery();
Response.Write(affectrows +" 條記錄受影響");

?3,ExecuteScalar() 這個方法不接受任何引數,僅僅返回查詢結果集中的第一行第一列,而忽略了其它的行和列,而且返回的是一個object型別,在使用之前必須先將它強制轉換為所需型別。如果返回的僅僅是一個單獨的資料元,則可以使用此方法來提高程式碼的效能。例:

string strCon="server=localhost;database=Northwind;Trusted_Connection=Yes;";
string strqry="select count(*) from Categories";
SqlConnection con=new SqlConnection(strCon);
con.Open();
SqlCommand cmd=con.CreateCommand();
int i=Convert.ToInt32(cmd.ExecuteScalar()); //必須強制轉換

 4,ExecuteXmlReader() 此方法用於XML操作,返回一個XmlReader物件,由於系統預設沒有引用 System.Xml名空間,因此在使用前必須前引入。例:

string strCon="server=localhost;database=Northwind;Trusted_Connection=Yes;";
SqlConnection con=new SqlConnection(strCon);
con.Open();
SqlCommand cmd = new SqlCommand("select * from Categories FOR XML AUTO, XMLDATA", con);
XmlReader xr=cmd.ExecuteXmlReader();
Response.Write(xr.AttributeCount);  //這裡獲取當前節點上的屬性個數
?

xr.Close();

 執行完畢之後,照樣要顯式地呼叫Close()方法,否則會丟擲異常。

 使用引數化的查詢

  先看一段SQL語句:select CategoryID,Description from Categories where CategoryID=? 其中的問號就是一個引數。但在使用的時候必須是帶有@字首的命名引數,因為.NET資料提供程式不支援這個通用的引數標記“?”.使用引數化的查詢可以大大地簡化程式設計,而且執行效率也比直接查詢字串要高,也更方便,很多情況下都需要更改查詢字串,這種方式就提供了方便,只需更改引數的值即可。例:

 string strCon="server=localhost;database=Northwind;Trusted_Connection=Yes;";
SqlConnection con=new SqlConnection(strCon);
con.Open();
string strqry="select * from Categories where CategoryID=@CategoryID"; //帶引數的查詢
SqlCommand cmd=new SqlCommand(strqry,con);
cmd.Parameters.Add("@CategoryID",SqlDbType.Int,4); //給引數賦於同資料庫中相同的型別
cmd.Parameters["@CategoryID"].Value="3"; //給引數賦值,可靈活改變
SqlDataReader r=cmd.ExecuteReader(); 
while(r.Read())
{
Response.Write(r.GetString(2)+"
"); //取出指定引數列的值
}
con.Close(); //切記關閉

 使用儲存過程進行查詢

 先看段儲存過程的形式:create procedure cateproc (@CategoryID int(4)) as select * from Categories where CategoryID=@CategoryID? return。

 這個是資料庫中的儲存過程實現方式,要在程式中呼叫儲存過程,一種方法是使用Command物件的 CommandType屬性來實現。CommandType有三個列舉值:Text,TableDirect,StoredProcedure。只需將CommandType屬性設為第三個值即可實現呼叫儲存過程。例:

 string strCon="server=localhost;database=Northwind;Trusted_Connection=Yes;";
SqlConnection con=new SqlConnection(strCon);
con.Open();
SqlCommand cmd=con.CreateCommand();
cmd.CommandText="cateproc";
cmd.CommandType=CommandType.StoredProcedure;
cmd.Parameters.Add("@CategoryID",SqlDbType.Int,4);
cmd.Parameters["CategoryID"].Value="2";
SqlDataReader r=cmd.ExecuteReader();
while(r.Read())
{
Response.Write(r.GetString(2)+"
");
}
con.Close();

 其實在程式中實現呼叫儲存過程的方式跟引數化查詢很類似,有點舊鞋翻新的味道。

 cmd.CommandType=CommandType.StoredProcedure;這種方式有個缺點,就是當要查詢的表,檢視或儲存過程的名稱中有特殊的字元(如空格)的話,則將無法識別。因此還有一種方式就是:

 cmd.CommandText="{Call cateproc(?)}"; //這裡是呼叫儲存過程,問號為引數
cmd.CommandType=CommandType.Text; //關鍵是這裡。

 設定命令執行超時

 命令超時是指Command物件在等待結果的時間,(預設為30秒)如果在30秒內沒執行查詢,則Command丟擲一個異常。也可以自己進行設定。例:cmd.CommandTimeout=60;

 取消執行查詢

 有時因某種原因,需要臨時取消命令的執行,可呼叫Command物件的Cancel()方法來退出執行,如果在未執行查詢之前,Cancel()將不做任何事。

相關文章