ADO.NET中的視圖和過濾器
作者:不詳
ADO.NET中有一層對象,用來建立任意資料源的抽象模型。其中包括 DataSet,DataTable,DataRow,DataView,DataRelation等等。
所有這些對象都定義在 System.Data名字空間。它們形成一個抽象模型,使得無論針對Windows Form,Web Form還是Web Service進行程式設計,都可以使用相同的程式設計接口。
在實際應用中,這些對象大多會對諸如SQL Server一類的關系型資料庫中的資料進行操作。但是,它們可以處理各種資料,而不管它的實體存儲媒介。
你可以使用 DataSet對象來打包和關聯各表中的資料,用DataTable類來處理表格類型的資料,而 DataRow對象可以處理表中某一行的資料。
這三個對象都是對資料進行打包,但有不同的邏輯聚合層次。 DataSet是 DataTable和其他的組合。而 DataTable是 DataRow和其他的組合。 DataRow是字段和其他的組合。但是這些對象中都沒有内建過濾和排序的功能。
ADO.NET提供了一些類來處理這個資料庫應用程式中的重要方面。在.Net Beta2中,這方面最重要的兩個對象就是 DataView和 DataViewManager。
注意:DataViewManager是Beta2中特有的。在Beta1中,相應的功能由DataSetView完成。
定制資料視圖
DataView類用來表示定制的 DataTable的視圖。 DataTable和 DataView的關系是遵循著名的設計模式--文檔/視圖模式,其中 DataTable是文檔,而 Dataview是視圖。
在任何時候,你都可以有多個基于相同資料的不同的視圖。更重要的是,你可以對每一個具有自己一套屬性、方法、事件的視圖作為獨立的對象進行處理。這也代表了相對ADO一個巨大的飛躍。
ADO Recordset可以定義過濾字元串。一旦你建立了該字元竄,隻有比對特定标準的資料才能夠進行讀寫。 Filter屬性的工作原理同動态WHERE子句很相似。它隻是簡單的在同一recordset對象視圖上隐藏了某些記錄。
在ADO中,你從沒有一個獨立的視圖對象。一個過濾過的recordset總是同一個對象,隻不過顯示出的記錄比它實際數量少一些而已。
如果你不需要同時處理一些不同的視圖,上述問題并不要緊。程式設計接口賦予了recordset既可以是表也可以是視圖的功能。但是在建立時,這不能同時發生。在某一特定時刻,recordset隻能是沒有過濾字元串的表或者是激活了過濾字元串的視圖。
Recordset的克隆提供了較好解決這個結構限制的方法。正如Clonation and the Case of Table Dolly, Part 1中所說的,克隆recordset相對開銷較少,因為它不複制資料,隻是複制recordset的基本構造。要處理同一資料兩個或兩個以上的視圖,你可以利用兩個或兩個以上克隆,各自有一套相應的過濾字元串。
圖一 在ADO中處理同一recordset不同的視圖
在ADO.NET中,你可以使用新型對象模型所提供的 DataView對象。ADO.NET的 DataView對象用來表示給定資料表的定制的視圖,但你可以像處理單獨的對象一樣處理它。 DataView對象保留了對表的一個引用并允許對它進行更新。
圖二 在ADO.NET中對同一資料表的不同視圖進行操作
功能上而言,使用ADO Recordset克隆與使用特殊的視圖對象完成的是同樣的功能,都是讓你實作過濾,對所選的資料行進行操作,并同時處理多個視圖。
深入DataView對象
DataView對象繼承了MarshalByValueComponent并實作了一組接口使之在資料綁定控件中可用。
Public Class DataView Inherits MarshalByValueComponent Implements IBindingList, IList, ICollection,IEnumerable, _ ITypedList, ISupportInitialize |
由MarshalByValueComponent派生的類是.NET遠端元件,可以通過值來列集--即序列化對象到目标應用程式域。(詳見以下關于.NET元件的更多細節)
DataView中的内容可以通過許多程式設計接口進行操作,包括集合,清單和枚舉器。IBindingList接口確定了該類提供所有用來支援複雜的和簡單的資料綁定的必要特征。
總的來說,DataView對象可以用來達到兩個目的。第一,視圖對于關聯DataTable對象和資料綁定控件中的DataSource域是很重要的。第二,它也對連接配接的DataTable提供了一層包裝,讓你能夠進行過濾,排序,編輯和浏覽。
DataView并不是唯一的可以通過傳值進行遠端操作的資料驅動類。DataSet和DataTable也具有同樣的能力,特别是在互操作的場景下。
建立DataView
public DataView(); public DataView(DataTable); |
DataView隻有同已經存在的、很可能是非空的DataTable對象連接配接後才可用。通常,這個連接配接在構造時就指定了。
DataView dv; dv = new DataView(theDataSet.Tables["Employees"]); |
但是,你也可以先建立一個新的視圖,然後再用Table屬性同表相關聯。
DataView dv = new DataView(); dv.Table = theDataSet.Tables["Employees"]; DataView構造函數使你由DataTable中得到一個DataView對象。如果需要,反之亦可。事實上,DataTable對象的DefaultView屬性傳回一個該表的DataView對象。 DataView dv = dt.DefaultView; |
一旦你有了 DataView對象, 你可以利用它的屬性來建立你希望使用者見到的資料行集。一般,你可以使用下列屬性:
- RowFilter
- Sort
前者可以定制視圖中可見資料應比對的規則。而後者通過表達式來進行排序。當然你可以使用這兩者的任意組合。
設定過濾 RowFilter是一個可讀寫的屬性,用來讀取和設定表過濾的表達式。
public virtual string RowFilter {get; set;} |
你可以用列名,邏輯和數字運算符和常量的任意合法組合組成表達式。以下是一些例子:
dv.RowFilter = "Country = 'USA'"; dv.RowFilter = "EmployeeID >5 AND Birthdate < #1/31/82#" dv.RowFilter = "Description LIKE '*product*'" |
讓我們來看一下過濾器的基本規則和運算符。
過濾字元串是表達式的邏輯連接配接。可以用AND,OR,NOT來連接配接成一個較短的表達式,也可以使用圓括号來組成子句,指定優先的運算。
通常包含列名的子句同字母、數字、日期或另一個列名進行比較。這裡,可以使用關系運算符和算術運算符,如>=, <, >, +, *, % (取模)等等。
如果要選取的行并不能友善地通過算術或邏輯運算符表達,你可以使用IN操作符。以下代碼顯示如何選取一個随機行:
dv.RowFilter = "employeeID IN (2,4,5)" |
你也可以使用通配符*和%,它們同LIKE運算符一起使用時顯得更有用。它們都表示任意數量的字元,可以互相替代使用。
請注意,如果在LIKE子句中已經有了*或%字元,你必須用方括号将其括起,以免歧義。如果很不幸,字元串中方括号本身也存在了,那麼它也必須用将本身括起。這樣,比對語句會如下所示:
dv.RowFilter = "Description LIKE '[[]*[]]product[[]*[]]" |
通配符隻允許在過濾字元串的開頭或結尾處使用,而不能在字元串中間出現。例如,下列語句會産生運作時錯誤:
dv.RowFilter = "Description LIKE 'prod*ct" |
字元串必須以單引号括起,而日期型必須以#符号括起。字元型值可以使用小數點和科學計數法。
RowFilter也支援聚合函數,如SUM, COUNT, MIN,MAX, and AVG。如果表中沒有資料行,那麼函數将傳回NULL。
在介紹RowFilter表達式的最後,讓我們讨論三個很便利的函數:Len,IIF和Substring。
正如其名,Len()傳回特定表達式的長度。該表達式可以是一個列名,也可以是其他合法的表達式。
Substring()傳回指定的表達式自特定位置開始,特定長度的字元子串。
我最喜歡用的是IIF(),它按照邏輯表達式的值有一到兩個值。IIF是IF-THEN-ELSE語句的緊湊表達。文法如下:
IIF(expression, if_true, if_false) |
通過該函數,可以建立非常複雜的過濾字元串。例如,假定你從SQL Server的Northwind資料庫中取得Employees表,下清單達式可以選出那些employeeID小于6且lastname為偶數個字元和employeeID大于6且lastname為奇數個字元的員工。
IIF(employeeID<6, Len(lastname) %2 =0, Len(lastname) %2 >0) |
下圖顯示了結果(樣品應用程式會在稍後讨論)
圖三 對Northwind中的表進行過濾
例子程式是一個Windows® Form應用程式,其中使用了兩個datagrid
控件來實作master/detail結構。一個grid在載入時生成,即在SQL
Server data adapter完成資料讀取工作之後。請注意,data
adapter是Beta 2中引入的,在Beta 1中相應的是SQLDataSetCommand類。
預排視圖
在上面的舉例中,datagrid必須負責預排視圖中的資料行,以便重新整理使用者界面。這個自動機制是.NET
資料綁定的産物。Datagrid是通過 DataSource屬性來擷取資料的資料綁定控件。DataView是一個可資料綁定的類,可建構 DataSource屬性的内容。
如果你想使用datagrid之外的另一個控件,應該怎麼辦呢?又如果你不想使用自動資料綁定呢?應該怎樣預排視圖中所選的資料行呢?
DataView的 Table屬性指向相應的資料表,但D ataTable并不儲存過濾資訊。是以,預排表中的資料注定是不可行的。雖然D ataTable和 DataView是緊密相聯的,但它們各自保持獨立,并執行獨立的功能。
以下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 |
正如前面說提到的, DataView是可枚舉的類,是以你可以安全的将它傳給For..Each語句。Count屬性存儲了視圖中資料行數,以便在For..Next循環中使用。
要通路視圖中某一行,可以使用 DataRowView類。 DataRowView可表示 DataRow的視圖,就像 DataView表達DataTable定制的視圖一樣。
總的來說,DataRow最多有四種狀态:default,original,current和proposed。這些狀态由 DataRowVersion枚舉類型設定,由 RowVersion屬性表達。
DataRow的視圖隻能是其中某一種狀态。
資料行的預設(default)版本隻有當其列在構造時設定了預設值時才有。而初始(original)版本是指在最後一次調用表的 AcceptChanges後,從數劇源中得到資料行或快照。目前(Current)版本是指目前的資料行,包括所有當時發生的更新。Proposed狀态隻存在于調用BeginEdit和EndEdit的編輯過程中。
可以通過通路 DataRow相同的文法通路 DataRowView。這裡最重要的屬性叫Item。
排序和其他便捷的特性
DataView支援Sort屬性,可以用來對視圖中的内容排序。Sort由用逗号分隔的列名表達式進行排序。通過在任何列名後加ASC或者DESC限定詞,可以使得字段按照上升或者下降的順序排列。如果沒有方向限定詞,預設順序為ASC。
DataView是記憶體中的對象,是以排序在本地進行,無需調用資料庫伺服器。
RowStateFilter是DataView另一有趣的屬性。它可以用任何預定義的标準來過濾DataTable中的内容。下表中是DataViewRowState枚舉類型的所有取值:
CurrentRows | 包括所有未更新的、新的和修改的資料行 |
Deleted | 所有自上次調用AcceptChanges後删除的資料行 |
ModifiedCurrent | 所有自上次調用AcceptChanges後修改過的資料行 |
ModifiedOriginal | 所有自上次調用AcceptChanges後original版本的資料行 |
New | 所有自上次調用AcceptChanges後新添加的行 |
OriginalRows | 傳回初始資料行,包含unchanged和deleted 的 |
Unchanged | 所有未更新的資料行 |
如果要操作非連接配接的資料,所有更新都在對 DataTable調用 AcceptChanges後生效。對單一行的更新在調用 DataRow的 AcceptChanges後生效。類似的,這些更新可以通過調用 DataTable或 DataRow對象的 RejectChanges來取消。
DataView對象還有一些屬性,如 AllowEdit,AllowDelete和 AllowNew,用來得到或設定是否允許更新的值。它們的預設值設為True,允許任何種類的更新。如果在标志設為False時,你想要完成相應的更新操作,會有一個運作時錯誤發生。
DataViewManager類
DataTable對象的 DefaultView屬性用來傳回一個 DataView對象,作為資料表中内容的預設視圖。它按照自然順序讀取資料并顯示表中所有的行,而不使用任何過濾。
theMasterGrid.DataSource = m_ds.Tables("Employees").DefaultView |
如果需要資料特定的視圖,你可以進行排序并/或對DefaultView對象直接進行過濾。
m_ds.Tables("Employees").DefaultView.Sort = "lastname" theMasterGrid.DataSource = m_ds.Tables("Employees").DefaultView |
DataViewManager類是用來存儲DataSet中所有表的視圖設定。
可以通過傳遞一個合法的非空的DataSet給類的構造函數來建立DataViewManager
Dim dvm As DataViewManager dvm = New DataViewManager(m_ds) |
也可以通過DataSet對象的DefaultViewManager屬性直接得到:
Dim dvm As DataViewManager = m_ds.DefaultViewManager |
重要的是DataViewManager類是同一個DataSet相關聯的。下面是另一種可行的方法:
Dim dvm As New DataViewManager() dvm.DataSet = m_ds |
DataViewManager最重要的屬性是DataViewSettings,一個DataViewSetting對象的集合。
Dim dvs As DataViewSetting dvs = dvm.DataViewSettings("Employees") dvs.Sort = "lastname" |
DataViewSetting對象包含了表視圖的參數資訊。當将資料綁定到對資料敏感的控件時,使用DataViewManager而不是DataSet或DataTable可以保留你的視圖設定(過濾和排序字段)
theMasterGrid.DataSource = dvm theMasterGrid.DataMember = "Employees" |
在這裡,視圖按照DataViewSetting中對Employees表指定的自動進行排序和過濾。換而言之,DataViewSetting類是對特定表的視圖的一種緩存。
下一步
上述例子程式用filter實作了master/detail結構。如果使用.NET中特有的資料綁定控件(如datagrid),能夠更好的達到這個目的。在以後的專欄中,我将論述記憶體中的資料關系,以及它們是如何影響master/detail結構的設計的。
對話:你是否需要控件或元件?
在.NET中有很多術語經常可以替代使用。這裡特别指出的是:類,元件,對象和控件。在此,我提供了一張表,來表述每個術語的恰當的含義。我們經常将它們當成同義詞。
需要牢記在心的是整個.NET架構是由類組成的。是以你從中得到的任何東西,首先,是一個類。在.NET環境中,控件群組件不是同一種類。至于對象,可以認為是運作着的.NET類的執行個體。
元件是一個特殊的類,它實作了Icomponent接口或派生于實作了Icomponent接口的類。
控件是提供了使用者界面功能的元件。在.NET架構中,可以找到兩類控件:用戶端的Windows Forms 控件和ASP.NET server 控件。
Icomponent接口包含在Idisposable接口中,并提供了一種确定的方法清除資源。
Public Interface IComponent Inherits IDisposable |
這種釋放資源的方法和标準的.NET垃圾收集器可以二者選一。通過實作Idisposable,你定義了一個Dispose方法。這樣通過程式設計,你可以顯式的釋放對象而無須等待垃圾收集器來處理。
.NET元件知道怎樣在應用程式域(application domain)中如何串聯。這有兩種方法:通過引用或通過值,基本功能分别内建于MarshalByRefComponet和MarshalByValueComponent類中。.NET component類,事實上,實作了Idisposable,但直接或間接繼承了上述兩個類中的一個。
應用程式域是一種輕量級程序。通過引用來列集對象意味着proxy/stub實體對會被建立并處理遠端調用。而通過值則意味着該對象的序列化的拷貝傳遞越過域的邊界。
控件是更特殊化的對象,它還提供了使用者界面元素。當然,一個控件總是一個component,但反之不一定成立。