1. 程式人生 > >Net編程 詳解DataTable用法【轉】

Net編程 詳解DataTable用法【轉】

security 操作 出錯 eight 按鈕 rdate sum 客戶 基本

http://www.diybloghome.com/article/16.html

DataTable表示一個與內存有關的數據表,可以使用工具欄裏面的控件拖放來創建和使用,也可以在編寫程序過程中根據需要獨立創建和使用,最常見的情況是作為DataSet的成員使用,在這種情況下就需要用在編程過程中根據需要動態創建數據表。

1 代碼創建DataTable數據表
通過添加對象的方式直接在DataSet中創建數據表,可以通過使用Add方法將DataTable添加到DataSet中,這種是使用控件的可視化添加DataTable的操作,那麽在代碼中怎麽來創建DataTable數據表呢?
在程序中創建DataTable對象可以使用相應的DataTable構造函數。創建一個表名為TableName的數據表,實現代碼如下所示:

  1. DataTable NewTable = new DataTable(TableName);

另外也可以通過以下方法創建DataTable對象:使用DataAdapter對象的Fill方法或FillSchema方法在DataSet中創建,這種方式都用於與數據庫相連接操作的情況下。實現代碼如下所示:

  1. //數據庫聯接字符串

  2. string connectionString =

  3. "Data Source=local;Initial Catalog=Northwind;Integrated Security=True;UserID=sa;Password=";

  4. //sql語句查詢

  5. string commandString = "Select * from Customers";

  6. // 創建SqlDataAdapter對象,並執行sql命令

  7. SqlDataAdapter dataAdapter = new SqlDataAdapter(commandString, connectionString);

  8. //創建數據集dataSet

  9. DataSet dataSet = new DataSet();

  10. //把數據表添加到數據集中

  11. DataTable dataTable = dataSet.Tables("Customers");

  12. //將數據填充到數據集中

  13. dataAdapter.Fill(dataSet,"Customers");

註意:將一個DataTable作為成員添加到一個DataSet的Tables集合中後,不能再將其添加到任何其他DataSet的表集合中。
使用DataTable構造函數初次創建DataTable時,是沒有架構(即結構,沒有列)的。沒有架構的DataTable數據表示沒有辦法使用的,因此要在使用這種DataTable數據表之前要定義表的架構,必須創建DataColumn對象並將其添加到表的Columns集合中。如何使用代碼創建Columns列,將在本章後面幾節中講到。
創建DataTable時,不需要為TableName屬性提供值,可以在其他時間指定該屬性,或者將其保留為空,這些都不影響DataTable的使用。應該註意的是在將一個沒有TableName值的表添加到DataSet中時,該表會得到一個從“Table”(表示Table0)開始遞增的默認名稱TableN。
以下示例創建DataTable對象的實例,並為其指定名稱“Customers”。 實現代碼如下所示:

  1. DataTable workTable = new DataTable("Customers");

以下代碼是將創建的DataTable實例Customers表添加到DataSet的Tables集合中。實現代碼如下所示:

  1. DataSet customers = new DataSet();

  2. DataTable customersTable = customers.Tables.Add("CustomersTable");

或者

  1. DataSet customers = new DataSet();

  2. DataTable customersTable = new DataTable(“Customers”)

  3. customers.Tables.Add(Customers);

2 用編程方式添加DataTable列
前面已經學過使用代碼創建DataTable,但是使用DataTable構造函數初次創建 DataTable時,是沒有架構(即結構,沒有列)的。要定義表的架構,必須創建DataColumn 對象並將其添加到表的Columns集合中。也可以為表定義主鍵列,並且可以創建Constraint約束對象並將其添加到表的Constraints約束集合中。
DataColumn類型表示了DataTable上的一列。總的來說,綁定到某個DataTable的所有DataColumn類型的集合就表示一個表。
DataTable包含了由表的Columns屬性引用的DataColumn對象的集合。這個列的集合與任何約束一起定義表的架構(即結構)。
通過使用DataColumn構造函數,或者通過調用表的Columns屬性的Add方法,可在表內創建DataColumn對象。Add方法將接受可選的ColumnName、DataType參數,並將創建新的DataColumn作為集合的成員。它還會接受現有的DataColumn對象並會將其添加到集合中,並會根據請求返回對所添加的DataColumn的引用。
以下示例向DataTable中添加了四列。實現代碼如下所示:

  1. DataTable workTable = new DataTable("Customers");

  2. DataColumn workCol = workTable.Columns.Add("CustID");

  3. workTable.Columns.Add("CustLName");

  4. workTable.Columns.Add("CustFName");

  5. workTable.Columns.Add("Purchases");

代碼說明:

CustID,CustLName,CustFName,Purchases:數據表Customers中的列名。

3 設置DataTable數據表的主鍵
數據庫開發的一個通常規則就是表至少得有一個列作為主鍵。主鍵約束用於惟一標識給定表中的一條記錄(行)。假設現在需要新建一個DataColumn列來表示EmpID字段並且要將這個列將作為表的主鍵,它必須有AllowDBNull和Unique屬性,實現代碼如下所示:

  1. DataTable workTable = new DataTable("Customers");

  2. DataColumn workCol = workTable.Columns.Add("CustID", typeof(Int32));

  3. workCol.AllowDBNull = false;

  4. workCol.Unique = true;

  5. workTable.Columns.Add("CustLName", typeof(String));

  6. workTable.Columns.Add("CustFName", typeof(String)));

  7. workTable.Columns.Add("Purchases", typeof(String)));

代碼說明:
示例中用於CustID列的屬性設置為不允許DBNull值並將值約束為唯一。但是,如果將CustID列定義為表的主鍵列,AllowDBNull屬性就會自動設置為false,並且Unique屬性會自動設置為true。


4 設置列的數據類型
通過上面8.4.2節的學習已經知道怎麽向新建的數據表中添加列了,那麽下面來看一看,怎麽為添加的列設置列的數據類型。數據類型是標明一列數據的數據類型屬性,根據不同的需要可以在DataTable數據表中建立不同的列,並可以為不同的列設置不同的數據類型,來滿足需要。繼續上面Customers表的例子,為創建的列添加數據類型:
實現代碼如下所示:

  1. //創建一個數據表Customers

  2. DataTable CustomersTable = new DataTable("Customers");

  3. //創建一個Int32類型名稱是CustID列,把這個列設置成主鍵,並且不允許為空,

  4. DataColumn CustomersCol = CustomersTable.Columns.Add("CustID", typeof(Int32));

  5. CustomersCol.AllowDBNull = false;

  6. CustomersCol.Unique = true;

  7. //創建三個String類型的列CustLName,CustFName,Purchases

  8. CustomersTable.Columns.Add("CustLName", typeof(String));

  9. CustomersTable.Columns.Add("CustFName", typeof(String)));

  10. CustomersTable.Columns.Add("Purchases", typeof(String)));

代碼說明:
示例中用於CustID列定義為表的主鍵列。CustID列指定的數據類型是Int32,CustLName列、CustFName列、Purchases列指定的數據類型都是String的列,當然也可以不設置列的數據類型,在這個時候DataColumn的DataType屬性默認為字符串類型,當然可以根據需要在創建列名時進行列數據類型的設置。

5 啟用Autoincrementing字段
在8.4.4節中學會了如何設置DataColumn列的數據類型,在設置完DataColumn列的數據類型以後,也可以像SQL-Server數據庫表一樣把某一列設置成自動遞增的。簡單地說,自動增加列可以確保當一個新行被添加到給定表時,可以基於當前的遞增步長值自動指定這個列的值。特別是某一列作為沒有重復值得主鍵的時候,這個功能就特別有用。在DataTable中這個功能可以用AutoIncrement(列是否將列的值自動遞增)、AutoIncrementSeed(起始值,種子值)和AutoIncrementStep(步長)屬性來控制。
下面是創建一個支持自動遞增的DataColumn列的例子。種子值用於標記列的起始值,步長值表示遞增時增加種子值的數值,代碼如下所示:

  1. // 創建一個新列

  2. DataColumn myColumn = new DataColumn();

  3. myColumn.ColumnName = " CustID ";

  4. myColumn.DataType = System.Type.GetType("System.Int32");

  5. // 設置自動遞增

  6. myColumn.AutoIncrement = true;

  7. myColumn.AutoIncrementSeed = 0;

  8. myColumn.AutoIncrementStep = 1;

代碼說明:
AutoIncrement:列是否將列的值自動遞增,true表示自動遞增,false表示不能自動遞增。
AutoIncrementSeed:起始值種子值,AutoIncrement屬性設置為true的列的起始值。
AutoIncrementStep:步長,遞增量,AutoIncrement屬性設置為true的列的步長。
創建一個數據類型為Int32的CustID列,為了能使這個字段的值實現自動增加的效果,把列AutoIncrement屬性設置為true;把列得種子值AutoIncrementSeed定為0,也就是從0開始計數;同時設置自動增加的步長AutoIncrementStep為1,每次增加一個。由於種子值被定為0,前面5個值應該是0、1、2、3和4。
可以往一個DataTable中添加這個DataColumn來測試一下。然後往這個表中添加一些新行,當然會自動轉儲CustID列中的值,代碼如下所示:

  1. //實現列自動增加功能

  2. protected void Button1_Click(object sender, EventArgs e)

  3. {

  4. //創建一個新的數據列,名稱:CustID ,數據類型:Int32.

  5. DataColumn myColumn = new DataColumn();

  6. myColumn.ColumnName = " CustID ";

  7. myColumn.DataType = System.Type.GetType("System.Int32");

  8. //把新創建的列設置自動增加,種子為0,增加步長為1.

  9. myColumn.AutoIncrement = true;

  10. myColumn.AutoIncrementSeed = 0;

  11. myColumn.AutoIncrementStep = 1;

  12. //把這個列添加到Customers表中.

  13. DataTable CustomersTable = new DataTable("Customers");

  14. myTable. CustomersTable.Add(myColumn);

  15. //添加20個新行.

  16. DataRow r;

  17. for (int i = 0; i < 20; i++)

  18. {

  19. r = CustomersTable.NewRow();

  20. CustomersTable.Rows.Add(r);

  21. }

  22. //顯示每一行的數據值.

  23. string temp = "";

  24. DataRowCollection rows = CustomersTable.Rows;

  25. for (int i = 0; i < CustomersTable.Rows.Count; i++)

  26. {

  27. DataRow currRow = rows;

  28. temp += currRow["CustID "] + " ";

  29. }

  30. //在Label1面顯示所有值

  31. Label1.Text = Label1.Text+temp;

  32. }

如果把上面代碼寫在建立的Web程序中,運行後點擊“Button”按鈕,就會得出初始值為0的步長為1的一系列數,

6 用編程方式添加DataTable行
在為DataTable定義了架構之後,也就是設置好了需要的列名以後,就可以可通過將DataRow對象添加到表的Rows集合中來將數據行添加到表中。與添加DataColumn類似,同樣可以通過使用DataRow構造函數,或者通過調用表的Rows屬性的Add方法,可在表內創建DataRow對象。
DataColumn對象集合表示了表的模式(Schema)。DataTable通過內部的DataColumnCollection類型保存表中所有列。相反,DataRow類型集合就表示表中的實際數據。這樣,如果Customers表中有10個記錄,就可以使用10個DataRow類型來表示它們。使用DataRow類的成員可以對表中的值進行插入、刪除、求值和操作操作。
創建一個DataRow數據行的對象,實現代碼如下所示:

  1. //創建一個Customers數據表

  2. DataTable CustomersTable = new DataTable("Customers ");

  3. //創建一個新的數據行

  4. DataRow arow = CustomersTable.NewRow();

  5. //設置行的值

  6. arow[ColumnName] = DataValue;

  7. //把數據行添加創建的Customers數據表中

  8. CustomersTable.Rows.Add(arow);

功能說明:
新建一行arow, 並給這行某一個列名付值為DataValue,最後把這一行添加到Customers表中。使用DataRow與使用DataColumn有些不同,因為不可以直接創建這個類型的實例,而是獲得一個來自給定DataTable的引用。例如,假設想往Customers表中添加新行,DataTable.NewRow()方法可以獲得下一空位,然後在上面填充每列的數據。
實現代碼如下所示:

  1. //創建一個數據表

  2. DataTable CustomersTable = new DataTable("Customers");

  3. //聲明數據表的行和列變量

  4. DataColumn column;

  5. DataRow row;

  6. //創建一個新列,設置列的數據列性和列名,並把這個新列添加到Customers表中

  7. column=new DataColumn();

  8. column.DataType = System.Type.GetType("System.Int32");

  9. column.ColumnName = " CustID ";

  10. CustomersTable.Columns.Add(column);

  11. //再創建一個新列

  12. column = new DataColumn();

  13. column.DataType = Type.GetType("System.String");

  14. column.ColumnName = " CustLName ";

  15. CustomersTable.Columns.Add(column);

  16. //創建新的一行並把這個行添加到Customers表中

  17. for(int i = 0; i < 10; i++)

  18. {

  19. row = CustomersTable.NewRow();

  20. row["CustID "] = i;

  21. row["CustLName "] = "item " + i.ToString();

  22. CustomersTable.Rows.Add(row);

  23. }

功能說明:
向CustomersTable表創建兩個列分別是CustID, CustLName,然後用循環的方式產生10行並附值,添加到CustomersTable表中。

7 操作DataTable:更新行
在前面一節裏面講了,怎麽樣添加一個DataRow行,那麽下面需要了解的關於DataTable的另一個方面就是怎樣用新值更新已有的DataRow行。下面開始介紹關於DataTable更新行的問題。
最常使用的一個方法就是先用Select()方法獲得符合給定過濾條件的行。一旦獲得這些DataRow,就對它們作相應的修改。例如,假定有一個新按鈕在被單擊後,搜索DataTable中所有EmployeeID列值為5的行。一旦標識這些項後,就可以把EmployeeID列對應值5改為6。實例代碼如下:

  1. //數據庫聯接字符串

  2. string connectionString =

  3. "Data Source=local;Initial Catalog=Northwind;Integrated Security=True;UserID=sa;Password=";

  4. //sql語句把Orders表中的數據信息都取出來

  5. string commandString = "Select * from Orders";

  6. SqlDataAdapter dataAdapter = new SqlDataAdapter(commandString, connectionString);

  7. DataSet dataSet = new DataSet( );

  8. //填充數據集

  9. dataAdapter.Fill(dataSet," Orders");

  10. //填充數據表

  11. DataTable dataTable = dataSet.Tables("Orders");

  12. // 建立一個用於過濾出EmployeeID列值為5的行的行的變量

  13. string filterStr = "EmployeeID=5";

  14. string strEmployeeID = null;

  15. // 查詢出來所有EmployeeID列值為5的行.

  16. DataRow[] EmployeeID= dataTable.Select(filterStr);

  17. // EmployeeID列對應值5改為6

  18. for(int i = 0; i < EmployeeID.Length; i++)

  19. {

  20. DataRow temp = EmployeeID ;

  21. strEmployeeID += temp["EmployeeID "] = 6;

  22. EmployeeID = temp;

  23. }

8 用編程方式刪除DataTable行
用於從DataTable對象中刪除DataRow對象的方法有兩種:DataRowCollection對象的 Remove方法和DataRow對象的Delete方法。
Remove方法和Delete方法都可以將DataTable的行DataRow刪除,但是前者是從DataRowCollection中刪除DataRow,而後者只將行標記為刪除。當應用程序調用AcceptChanges方法時,才會發生實際的刪除。通過使用Delete,可以在實際刪除之前先以編程方式檢查哪些行標記為刪除。如果將行標記為刪除,其RowState屬性會設置為Deleted。
在將DataSet或DataTable與DataAdapter和關系型數據源一起使用時,用DataRow的 Delete方法移除行。Delete方法只是在DataSet或DataTable中將行標記為Deleted,而不會移除它。而DataAdapter在遇到標記為Deleted的行時,會執行其DeleteCommand方法以在數據源中刪除該行。然後,就可以用AcceptChanges方法永久移除該行。如果使用Remove 刪除該行,則該行將從表中完全移除,但DataAdapter不會在數據源中刪除該行。
DataRowCollection的Remove方法采用DataRow作為參數,並將其從集合中移除,實現代碼如下所示:

  1. DataTable CustomersTable = new DataTable("Customers ");

  2. DataRow workRow = empTable.NewRow();

  3. CustomersTable.Rows.Remove(workRow);

作為對比,以下示例演示了如何調用DataRow上的Delete方法來將其RowState改為Deleted。實現代碼如下所示:

  1. DataTable CustomersTable = new DataTable("Customers ");

  2. DataRow workRow = empTable.NewRow();

  3. workRow.Delete();

如果將行標記為刪除,並且調用DataTable對象的AcceptChanges方法,該行就會從DataTable中移除。相比之下,如果調用RejectChanges,行的RowState就會恢復到被標記為Deleted之前的狀態。


9 向DataTable填充數據
ADO.NET中的DataSet是數據的內存駐留表示形式,它提供了獨立於數據源的一致關系編程模型。DataSet表示整個數據集,其中包含表、約束和表之間的關系。由於DataSet獨立於數據源,DataSet可以包含應用程序本地的數據,也可以包含來自多個數據源的數據。與現有數據源的交互通過DataAdapter來控制。
DataAdapter的SelectCommand屬性是一個Command對象,用於從數據源中檢索數據。DataAdapter的InsertCommand、UpdateCommand和DeleteCommand屬性也是Command對象,用於按照對DataSet中數據的修改來管理對數據源中數據的更新。
DataAdapter的Fill方法用於使用DataAdapter的SelectCommand的結果來填充DataSet。Fill將要填充的DataSet和DataTable對象(或要使用從SelectCommand中返回的行來填充的DataTable的名稱)作為它的參數。
Fill方法使用DataReader對象來隱式地返回用於在DataSet中創建表的列名稱和類型以及用於填充DataSet中的表行的數據。表和列僅在不存在時才創建;否則,Fill將使用現有的DataSet架構。
以下代碼示例使用SqlDataAdapter對象獲取Microsoft SQL Server Northwind數據庫的數據填充到Customers數據集中的Customers表。
實現代碼如下所示:

  1. //使用DataAdapter的Fill方法向Customers表中填充數據

  2. string connectionString =

  3. "Data Source= local;Initial Catalog=Northwind;Integrated Security=True; UserID=sa;Password=";

  4. string commandString = "Select * from Customers";

  5. SqlDataAdapter dataAdapter = new SqlDataAdapter(commandString, connectionString);

  6. DataSet customers = new DataSet();

  7. dataAdapter.Fill(customers, "Customers");

10 在DataTable實現類似sql語句的查詢功能
在脫離數據庫的DataSet實現類似sql語句的查詢。
也就是(select ... from tablename where ...)這樣的功能。
將從DataSet中查詢出來的數據以行的形式保存到arow中。
實現代碼如下所示:

  1. DataSet ds= new DataSet();

  2. DataRow[] arow = ds.Tables[TableName].Select("" + ColumnsName+ "=‘" + DataValue + "‘");

代碼說明:
TableName:數據集ds中的表名。
ColumnsName:TableName表中的某一列名。
DataValue:和列名對應參數的值。
功能說明:
上面語句功能相當於sql語句中的查詢語句

  1. Select * From TableName where ColumnsName= DataValue


11 DataTable中數據記錄的統計
在使用數據庫時例如SQL-Server、Oracle這些數據庫,可以輕松的通過統計和計算函數例如Sum、Aver、Count等統計或計算出相關結果,那麽,在已經把數據檢索出來的DataTable數據集中能否同樣使用呢?在程序中根據需要動態創建的DataTable是沒有辦法使用sql語句進行查詢統計的,例如沒有辦法使用Select語句來獲取查詢統計結果。那麽在DataTable中怎麽來進行統計呢?


本節將介紹一個簡單的方法,可以輕松的獲得DataTable中的記錄統計結果。這個簡單的方法就是調用功能強大的DataTable的函數Compute。
函數如下:

  1. public object Compute(string expression,string filter)


參數說明:
qexpression 參數需要聚合函數,要計算的表達式字符串,基本上類似於Sql Server中的統計表達式。例如,以下是合法表達式:Count(ID)。
qfilter:統計的過濾字符串,只有滿足這個過濾條件的記錄才會被統計,也就是確定在表達式中使用哪些行。
以下示例,以SQL-Server數據庫中的Northwind數據庫中的Orders數據表,描述訂單信息,包含字段為:訂單號(OrderID)、客戶編號(CustomerID)、職工編號(EmployeeID)、訂貨時間(OrderDate)、船號(ShipVia)、運費(Freight)。
1.統計所有職工編號EmployeeID為5的數量:

  1. table.Compute("Count(*)","EmployeeID=5");

2.統計所有運費Freight中運費大於100的個數

  1. table.Compute("Count(*)","Freight>100‘");

3.統計運費Freight的平均值

  1. table.Compute("Aver(Freight)","true");

4.統計職工編號EmployeeID為5的運費總額:

  1. table.Compute("Sum(Freight)"," EmployeeID=5");

以上都是計算每一列的合計,那麽要添加一行求合計可以使用下面的方法:

  1. //創建一個數據集

  2. DataSet customers = new DataSet();

  3. //在數據集中添加一個名稱為CustomersTable的數據表

  4. DataTable customersTable = customers.Tables.Add("CustomersTable");

  5. //創建一個新行並添加到CustomersTable數據表中

  6. DataRow dataRow = new DataRow();

  7. dataRow= customersTable.NewRow();

  8. //然後就是統計了,聲明連個int 變量i:循環變量, colCnt: customersTable表中的列數

  9. int i ;

  10. int colCnt ;

  11. colCnt = customersTable.Cols.Count;

  12. for( i=0 ;i< colCnt -1;i++)

  13. //求第i列的和並且結果賦值給新的dataRow數據行

  14. dataRow(i)= customersTable.Compute("Sum("+i.ToString()+")","true");

  15. //把數據行添加到customersTable數據表中

  16. customersTable.Rows.Add(dataRow);

12 DataTable和xml的一些應用總結
在ASP.NET2.0中ADO.NET和XML結合的很緊密,第7章中已經詳細介紹了在ADO.NET中的數據通過DataSet很容易存取在XML中的,那麽在DataTable能像在DataSet中那樣操作XML數據文件嗎?答案是可以的,下面就是一個將DataTable中的數據寫入到XML文件中去的操作,實例代碼如下所示:

  1. //創建一個Customers數據表

  2. DataTable dt = new DataTable("Customers");

  3. //添加兩列CustID,CustLName數據類型分別為Int32,String,並添加到數據表中

  4. DataColumn dc1 = new DataColumn("CustID", Type.GetType("System.Int32"));

  5. DataColumn dc2 = new DataColumn("CustLName", Type.GetType("System.String"));

  6. dt.Columns.Add(dc1);

  7. dt.Columns.Add(dc2);

  8. //利用循環創建數據行並賦值,添加到數據表中

  9. for (int i = 0; i < al.Count; i++)

  10. {

  11. DataRow dr = dt.NewRow();

  12. DS_Option dso = (DS_Option)al;

  13. dr["CustID"] = dso.ID;

  14. dr["CustLName"] = dso.Name;

  15. dt.Rows.Add(dr);

  16. }

  17. //將數據表中的數據寫入到XML文件中

  18. string xmlstr;

  19. System.IO.StringWriter writer = new System.IO.StringWriter();

  20. dt.WriteXml(writer);

  21. xmlstr = writer.ToString();

  22. return xmlstr;

有了向XML文件寫數據的操作,當然也可以從XML文件中讀取數據信息到DataTable數據表中,方法累也類似,但要先建立好打DataTable的結構,不然會出錯。
實例代碼如下所示:

    1. string tbxml = xmlinfo;

    2. DataTable dt = new DataTable("Customers ");

    3. DataColumn dc1 = new DataColumn("CustID", Type.GetType("System.Int32"));

    4. DataColumn dc2 = new DataColumn("CustLName", Type.GetType("System.String"));

    5. dt.Columns.Add(dc1);

    6. dt.Columns.Add(dc2);

    7. System.IO.StringReader reader = new System.IO.StringReader(tbxml);

    8. dt.ReadXml(reader);

Net編程 詳解DataTable用法【轉】