定制數據視圖
DataView類(lèi)用來(lái)表示定制的DataTable的視圖。DataTable和DataView的關(guān)系是遵循著(zhù)名的設計模式--文檔/視圖模式,其中DataTable是文檔,而Dataview是視圖。
在任何時(shí)候,你都可以有多個(gè)基于相同數據的不同的視圖。更重要的是,你可以對每一個(gè)具有自己一套屬性、方法、事件的視圖作為獨立的對象進(jìn)行處理。這也代表了相對ADO一個(gè)巨大的飛躍。
創(chuàng )建DataView public DataView();
public DataView(DataTable);
DataView只有同已經(jīng)存在的、很可能是非空的DataTable對象連接后才可用。通常,這個(gè)連接在構造時(shí)就指定了。
DataView dv;
dv = new DataView(theDataSet.Tables["Employees"]);
但是,你也可以先創(chuàng )建一個(gè)新的視圖,然后再用Table屬性同表相關(guān)聯(lián)。
DataView dv = new DataView();
dv.Table = theDataSet.Tables["Employees"];
DataView構造函數使你由DataTable中得到一個(gè)DataView對象。如果需要,反之亦可。事實(shí)上,DataTable對象的DefaultView屬性返回一個(gè)該表的DataView對象。
DataView dv = dt.DefaultView;
一旦你有了DataView對象, 你可以利用它的屬性來(lái)建立你希望用戶(hù)見(jiàn)到的數據行集。一般,你可以使用下列屬性:
RowFilter
Sort
前者可以定制視圖中可見(jiàn)數據應匹配的規則。而后者通過(guò)表達式來(lái)進(jìn)行排序。當然你可以使用這兩者的任意組合。
設置過(guò)濾 RowFilter是一個(gè)可讀寫(xiě)的屬性,用來(lái)讀取和設置表過(guò)濾的表達式。
public virtual string RowFilter {get; set;}
你可以用列名,邏輯和數字運算符和常量的任意合法組合組成表達式。以下是一些例子: dv.RowFilter = "Country = 'USA'";
dv.RowFilter = "EmployeeID >5 AND Birthdate < #1/31/82#"
dv.RowFilter = "Description LIKE '*product*'"
讓我們來(lái)看一下過(guò)濾器的基本規則和運算符。
過(guò)濾字符串是表達式的邏輯連接??梢杂肁ND,OR,NOT來(lái)連接成一個(gè)較短的表達式,也可以使用圓括號來(lái)組成子句,指定優(yōu)先的運算。
通常包含列名的子句同字母、數字、日期或另一個(gè)列名進(jìn)行比較。這里,可以使用關(guān)系運算符和算術(shù)運算符,如>=, <, >, +, *, % (取模)等等。
如果要選取的行并不能方便地通過(guò)算術(shù)或邏輯運算符表達,你可以使用IN操作符。以下代碼顯示如何選取一個(gè)隨機行:
dv.RowFilter = "employeeID IN (2,4,5)"
你也可以使用通配符*和%,它們同LIKE運算符一起使用時(shí)顯得更有用。它們都表示任意數量的字符,可以相互替代使用。
請注意,如果在LIKE子句中已經(jīng)有了*或%字符,你必須用方括號將其括起,以免歧義。如果很不幸,字符串中方括號本身也存在了,那么它也必須用將本身括起。這樣,匹配語(yǔ)句會(huì )如下所示:
dv.RowFilter = "Description LIKE '[[]*[]]product[[]*[]]"
通配符只允許在過(guò)濾字符串的開(kāi)頭或結尾處使用,而不能在字符串中間出現。例如,下列語(yǔ)句會(huì )產(chǎn)生運行時(shí)錯誤:
dv.RowFilter = "Description LIKE 'prod*ct"
字符串必須以單引號括起,而日期型必須以#符號括起。字符型值可以使用小數點(diǎn)和科學(xué)計數法。
RowFilter也支持聚合函數,如SUM, COUNT, MIN,MAX, and AVG。如果表中沒(méi)有數據行,那么函數將返回NULL。
在介紹RowFilter表達式的最后,讓我們討論三個(gè)很便利的函數:Len,IIF和Substring。
正如其名,Len()返回特定表達式的長(cháng)度。該表達式可以是一個(gè)列名,也可以是其他合法的表達式。
Substring()返回指定的表達式自特定位置開(kāi)始,特定長(cháng)度的字符子串。
我最喜歡用的是IIF(),它按照邏輯表達式的值有一到兩個(gè)值。IIF是IF-THEN-ELSE語(yǔ)句的緊湊表達。語(yǔ)法如下:
IIF(expression, if_true, if_false)
通 過(guò)該函數,可以建立非常復雜的過(guò)濾字符串。例如,假定你從SQL Server的Northwind數據庫中取得Employees表,下列表達式可以選出那些employeeID小于6且lastname為偶數個(gè)字符 和employeeID大于6且lastname為奇數個(gè)字符的員工。
IIF(employeeID<6, Len(lastname) %2 =0, Len(lastname) %2 >0)
預排視圖
在上面的舉例中,datagrid必須負責預排視圖中的數據行,以便刷新用戶(hù)界面。這個(gè)自動(dòng)機制是.NET
數據綁定的產(chǎn)物。Datagrid是通過(guò)DataSource屬性來(lái)獲取數據的數據綁定控件。DataView是一個(gè)可數據綁定的類(lèi),可構建DataSource屬性的內容。
如果你想使用datagrid之外的另一個(gè)控件,應該怎么辦呢?又如果你不想使用自動(dòng)數據綁定呢?應該怎樣預排視圖中所選的數據行呢?
DataView的Table屬性指向相應的數據表,但DataTable并不保存過(guò)濾信息。所以,預排表中的數據注定是不可行的。雖然DataTable和DataView是緊密相聯(lián)的,但它們各自保持獨立,并執行獨立的功能。
以下Visual Basic .NET代碼段顯示了如何遍歷視圖中所有的數據行,并加入到listbox中。
Dim dv As New DataView()
dv = ds.Tables("Employees").DefaultView
dv.RowFilter = "employeeid >5"
ListBox1.Items.Clear()
Dim buf As String
Dim dr As DataRowView
For Each dr In dv
buf = ""
buf &= dr("lastname").ToString()& ", " & dr("firstName").ToString()
ListBox1.Items.Add(buf)
Next
正如前面說(shuō)提到的,DataView是可枚舉的類(lèi),因此你可以安全的將它傳給For..Each語(yǔ)句。Count屬性存儲了視圖中數據行數,以便在For..Next循環(huán)中使用。
要訪(fǎng)問(wèn)視圖中某一行,可以使用DataRowView類(lèi)。DataRowView可表示DataRow的視圖,就像DataView表達DataTable定制的視圖一樣。
總的來(lái)說(shuō),DataRow最多有四種狀態(tài):default,original,current和proposed。這些狀態(tài)由DataRowVersion枚舉類(lèi)型設置,由RowVersion屬性表達。
DataRow的視圖只能是其中某一種狀態(tài)。
數據行的默認(default)版本只有當其列在構造時(shí)設定了默認值時(shí)才有。而初始(original)版本是指在最后一次調用表的AcceptChanges后,從數劇源中得到數據行或快照。當前(Current)版本是指當前的數據行,包括所有當時(shí)發(fā)生的更新。Proposed狀態(tài)只存在于調用BeginEdit和EndEdit的編輯過(guò)程中。
可以通過(guò)訪(fǎng)問(wèn)DataRow相同的語(yǔ)法訪(fǎng)問(wèn)DataRowView。這里最重要的屬性叫Item。
排序和其他便捷的特性
DataView支持Sort屬性,可以用來(lái)對視圖中的內容排序。Sort由用逗號分隔的列名表達式進(jìn)行排序。通過(guò)在任何列名后加ASC或者DESC限定詞,可以使得字段按照上升或者下降的順序排列。如果沒(méi)有方向限定詞,默認順序為ASC。
DataView是內存中的對象,所以排序在本地進(jìn)行,無(wú)需調用數據庫服務(wù)器。
RowStateFilter是DataView另一有趣的屬性。它可以用任何預定義的標準來(lái)過(guò)濾DataTable中的內容。下表中是DataViewRowState枚舉類(lèi)型的所有取值: CurrentRows包括所有未更新的、新的和修改的數據行
Deleted所有自上次調用AcceptChanges后刪除的數據行
ModifiedCurrent所有自上次調用AcceptChanges后修改過(guò)的數據行
ModifiedOriginal所有自上次調用AcceptChanges后original版本的數據行
New所有自上次調用AcceptChanges后新添加的行
OriginalRows返回初始數據行,包含unchanged和deleted 的
Unchanged所有未更新的數據行
如果要操作非連接的數據,所有更新都在對DataTable調用AcceptChanges后生效。對單一行的更新在調用DataRow的AcceptChanges后生效。類(lèi)似的,這些更新可以通過(guò)調用DataTable或DataRow對象的RejectChanges來(lái)取消。
DataView對象還有一些屬性,如AllowEdit,AllowDelete和AllowNew,用來(lái)得到或設定是否允許更新的值。它們的默認值設為T(mén)rue,允許任何種類(lèi)的更新。如果在標志設為False時(shí),你想要完成相應的更新操作,會(huì )有一個(gè)運行時(shí)錯誤發(fā)生。
DataViewManager類(lèi)
DataTable對象的DefaultView屬性用來(lái)返回一個(gè)DataView對象,作為數據表中內容的默認視圖。它按照自然順序讀取數據并顯示表中所有的行,而不使用任何過(guò)濾。
theMasterGrid.DataSource = m_ds.Tables("Employees").DefaultView
如果需要數據特定的視圖,你可以進(jìn)行排序并/或對DefaultView對象直接進(jìn)行過(guò)濾。
m_ds.Tables("Employees").DefaultView.Sort = "lastname"
theMasterGrid.DataSource = m_ds.Tables("Employees").DefaultView
DataViewManager類(lèi)是用來(lái)存儲DataSet中所有表的視圖設置。
可以通過(guò)傳遞一個(gè)合法的非空的DataSet給類(lèi)的構造函數來(lái)創(chuàng )建DataViewManager
Dim dvm As DataViewManager
dvm = New DataViewManager(m_ds)
也可以通過(guò)DataSet對象的DefaultViewManager屬性直接得到:
Dim dvm As DataViewManager = m_ds.DefaultViewManager
重要的是DataViewManager類(lèi)是同一個(gè)DataSet相關(guān)聯(lián)的。下面是另一種可行的方法:
Dim dvm As New DataViewManager()
dvm.DataSet = m_ds
DataViewManager最重要的屬性是DataViewSettings,一個(gè)DataViewSetting對象的集合。
Dim dvs As DataViewSetting
dvs = dvm.DataViewSettings("Employees")
dvs.Sort = "lastname"
DataViewSetting對象包含了表視圖的參數信息。當將數據綁定到對數據敏感的控件時(shí),使用DataViewManager而不是DataSet或DataTable可以保留你的視圖設置(過(guò)濾和排序字段)
theMasterGrid.DataSource = dvm
theMasterGrid.DataMember = "Employees"
在這里,視圖按照DataViewSetting中對Employees表指定的自動(dòng)進(jìn)行排序和過(guò)濾。換而言之,DataViewSetting類(lèi)是對特定表的視圖的一種緩存。