明輝手游網(wǎng)中心:是一個(gè)免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺(tái)!

ADO.NET中的視圖與過(guò)濾器

[摘要]ADO.NET中有一層對(duì)象,用來(lái)創(chuàng)建任意數(shù)據(jù)源的抽象模型。其中包括DataSet,DataTable,DataRow,DataView,DataRelation等等。所有這些對(duì)象都定義在System.Data名字空間。它們形成一個(gè)抽象模型,使得無(wú)論針對(duì)Windows Form,Web Form還是...
ADO.NET中有一層對(duì)象,用來(lái)創(chuàng)建任意數(shù)據(jù)源的抽象模型。其中包括DataSet,DataTable,DataRow,DataView,DataRelation等等。

所有這些對(duì)象都定義在System.Data名字空間。它們形成一個(gè)抽象模型,使得無(wú)論針對(duì)Windows Form,Web Form還是Web Service進(jìn)行編程,都可以使用相同的編程接口。
在實(shí)際應(yīng)用中,這些對(duì)象大多會(huì)對(duì)諸如SQL Server一類(lèi)的關(guān)系型數(shù)據(jù)庫(kù)中的數(shù)據(jù)進(jìn)行操作。但是,它們可以處理各種數(shù)據(jù),而不管它的物理存儲(chǔ)媒介。
你可以使用DataSet對(duì)象來(lái)打包和關(guān)聯(lián)各表中的數(shù)據(jù),用DataTable類(lèi)來(lái)處理表格類(lèi)型的數(shù)據(jù),而DataRow對(duì)象可以處理表中某一行的數(shù)據(jù)。
這三個(gè)對(duì)象都是對(duì)數(shù)據(jù)進(jìn)行打包,但有不同的邏輯聚合層次。DataSetDataTable和其他的組合。而DataTableDataRow和其他的組合。DataRow是字段和其他的組合。但是這些對(duì)象中都沒(méi)有內(nèi)建過(guò)濾和排序的功能。
ADO.NET提供了一些類(lèi)來(lái)處理這個(gè)數(shù)據(jù)庫(kù)應(yīng)用程序中的重要方面。在.Net Beta2中,這方面最重要的兩個(gè)對(duì)象就是DataViewDataViewManager。
注意:DataViewManager是Beta2中特有的。在Beta1中,相應(yīng)的功能由DataSetView完成。

定制數(shù)據(jù)視圖
DataView類(lèi)用來(lái)表示定制的DataTable的視圖。DataTableDataView的關(guān)系是遵循著名的設(shè)計(jì)模式--文檔/視圖模式,其中DataTable是文檔,而Dataview是視圖。
在任何時(shí)候,你都可以有多個(gè)基于相同數(shù)據(jù)的不同的視圖。更重要的是,你可以對(duì)每一個(gè)具有自己一套屬性、方法、事件的視圖作為獨(dú)立的對(duì)象進(jìn)行處理。這也代表了相對(duì)ADO一個(gè)巨大的飛躍。
ADO Recordset可以定義過(guò)濾字符串。一旦你建立了該字符竄,只有匹配特定標(biāo)準(zhǔn)的數(shù)據(jù)才能夠進(jìn)行讀寫(xiě)。Filter屬性的工作原理同動(dòng)態(tài)WHERE子句很相似。它只是簡(jiǎn)單的在同一recordset對(duì)象視圖上隱藏了某些記錄。
在ADO中,你從沒(méi)有一個(gè)獨(dú)立的視圖對(duì)象。一個(gè)過(guò)濾過(guò)的recordset總是同一個(gè)對(duì)象,只不過(guò)顯示出的記錄比它實(shí)際數(shù)量少一些而已。
如果你不需要同時(shí)處理一些不同的視圖,上述問(wèn)題并不要緊。編程接口賦予了recordset既可以是表也可以是視圖的功能。但是在創(chuàng)建時(shí),這不能同時(shí)發(fā)生。在某一特定時(shí)刻,recordset只能是沒(méi)有過(guò)濾字符串的表或者是激活了過(guò)濾字符串的視圖。
Recordset的克隆提供了較好解決這個(gè)結(jié)構(gòu)限制的方法。正如Clonation and the Case of Table Dolly, Part 1中所說(shuō)的,克隆recordset相對(duì)開(kāi)銷(xiāo)較少,因?yàn)樗粡?fù)制數(shù)據(jù),只是復(fù)制recordset的基本構(gòu)造。要處理同一數(shù)據(jù)兩個(gè)或兩個(gè)以上的視圖,你可以利用兩個(gè)或兩個(gè)以上克隆,各自有一套相應(yīng)的過(guò)濾字符串。


圖一 在ADO中處理同一recordset不同的視圖
在ADO.NET中,你可以使用新型對(duì)象模型所提供的DataView對(duì)象。ADO.NET的DataView對(duì)象用來(lái)表示給定數(shù)據(jù)表的定制的視圖,但你可以像處理單獨(dú)的對(duì)象一樣處理它。DataView對(duì)象保留了對(duì)表的一個(gè)引用并允許對(duì)它進(jìn)行更新。


圖二 在ADO.NET中對(duì)同一數(shù)據(jù)表的不同視圖進(jìn)行操作
功能上而言,使用ADO Recordset克隆與使用特殊的視圖對(duì)象完成的是同樣的功能,都是讓你實(shí)現(xiàn)過(guò)濾,對(duì)所選的數(shù)據(jù)行進(jìn)行操作,并同時(shí)處理多個(gè)視圖。

深入DataView對(duì)象
DataView對(duì)象繼承了MarshalByValueComponent并實(shí)現(xiàn)了一組接口使之在數(shù)據(jù)綁定控件中可用。
Public Class DataView
Inherits MarshalByValueComponent
Implements IBindingList, IList, ICollection,IEnumerable, _
ITypedList, ISupportInitialize

由MarshalByValueComponent派生的類(lèi)是.NET遠(yuǎn)程組件,可以通過(guò)值來(lái)列集--即序列化對(duì)象到目標(biāo)應(yīng)用程序域。(詳見(jiàn)以下關(guān)于.NET組件的更多細(xì)節(jié))
DataView中的內(nèi)容可以通過(guò)許多編程接口進(jìn)行操作,包括集合,列表和枚舉器。IBindingList接口確保了該類(lèi)提供所有用來(lái)支持復(fù)雜的和簡(jiǎn)單的數(shù)據(jù)綁定的必要特征。
總的來(lái)說(shuō),DataView對(duì)象可以用來(lái)達(dá)到兩個(gè)目的。第一,視圖對(duì)于關(guān)聯(lián)DataTable對(duì)象和數(shù)據(jù)綁定控件中的DataSource域是很重要的。第二,它也對(duì)連接的DataTable提供了一層包裝,讓你能夠進(jìn)行過(guò)濾,排序,編輯和瀏覽。
DataView并不是唯一的可以通過(guò)傳值進(jìn)行遠(yuǎn)程操作的數(shù)據(jù)驅(qū)動(dòng)類(lèi)。DataSet和DataTable也具有同樣的能力,特別是在互操作的場(chǎng)景下。

創(chuàng)建DataView
public DataView();
public DataView(DataTable);

DataView只有同已經(jīng)存在的、很可能是非空的DataTable對(duì)象連接后才可用。通常,這個(gè)連接在構(gòu)造時(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構(gòu)造函數(shù)使你由DataTable中得到一個(gè)DataView對(duì)象。如果需要,反之亦可。事實(shí)上,DataTable對(duì)象的DefaultView屬性返回一個(gè)該表的DataView對(duì)象。
DataView dv = dt.DefaultView;

一旦你有了DataView對(duì)象, 你可以利用它的屬性來(lái)建立你希望用戶見(jiàn)到的數(shù)據(jù)行集。一般,你可以使用下列屬性:
  • RowFilter
  • Sort
前者可以定制視圖中可見(jiàn)數(shù)據(jù)應(yīng)匹配的規(guī)則。而后者通過(guò)表達(dá)式來(lái)進(jìn)行排序。當(dāng)然你可以使用這兩者的任意組合。

設(shè)置過(guò)濾 RowFilter是一個(gè)可讀寫(xiě)的屬性,用來(lái)讀取和設(shè)置表過(guò)濾的表達(dá)式。
public virtual string RowFilter {get; set;}

你可以用列名,邏輯和數(shù)字運(yùn)算符和常量的任意合法組合組成表達(dá)式。以下是一些例子:
dv.RowFilter = "Country = 'USA'";
dv.RowFilter = "EmployeeID >5 AND Birthdate < #1/31/82#"
dv.RowFilter = "Description LIKE '*product*'"

讓我們來(lái)看一下過(guò)濾器的基本規(guī)則和運(yùn)算符。
過(guò)濾字符串是表達(dá)式的邏輯連接。可以用AND,OR,NOT來(lái)連接成一個(gè)較短的表達(dá)式,也可以使用圓括號(hào)來(lái)組成子句,指定優(yōu)先的運(yùn)算。
通常包含列名的子句同字母、數(shù)字、日期或另一個(gè)列名進(jìn)行比較。這里,可以使用關(guān)系運(yùn)算符和算術(shù)運(yùn)算符,如>=, <, >, +, *, % (取模)等等。
如果要選取的行并不能方便地通過(guò)算術(shù)或邏輯運(yùn)算符表達(dá),你可以使用IN操作符。以下代碼顯示如何選取一個(gè)隨機(jī)行:
dv.RowFilter = "employeeID IN (2,4,5)"

你也可以使用通配符*和%,它們同LIKE運(yùn)算符一起使用時(shí)顯得更有用。它們都表示任意數(shù)量的字符,可以相互替代使用。
請(qǐng)注意,如果在LIKE子句中已經(jīng)有了*或%字符,你必須用方括號(hào)將其括起,以免歧義。如果很不幸,字符串中方括號(hào)本身也存在了,那么它也必須用將本身括起。這樣,匹配語(yǔ)句會(huì)如下所示:
dv.RowFilter = "Description LIKE '[[]*[]]product[[]*[]]"

通配符只允許在過(guò)濾字符串的開(kāi)頭或結(jié)尾處使用,而不能在字符串中間出現(xiàn)。例如,下列語(yǔ)句會(huì)產(chǎn)生運(yùn)行時(shí)錯(cuò)誤:
dv.RowFilter = "Description LIKE 'prod*ct"

字符串必須以單引號(hào)括起,而日期型必須以#符號(hào)括起。字符型值可以使用小數(shù)點(diǎn)和科學(xué)計(jì)數(shù)法。
RowFilter也支持聚合函數(shù),如SUM, COUNT, MIN,MAX, and AVG。如果表中沒(méi)有數(shù)據(jù)行,那么函數(shù)將返回NULL。
在介紹RowFilter表達(dá)式的最后,讓我們討論三個(gè)很便利的函數(shù):Len,IIF和Substring。
正如其名,Len()返回特定表達(dá)式的長(zhǎng)度。該表達(dá)式可以是一個(gè)列名,也可以是其他合法的表達(dá)式。
Substring()返回指定的表達(dá)式自特定位置開(kāi)始,特定長(zhǎng)度的字符子串。
我最喜歡用的是IIF(),它按照邏輯表達(dá)式的值有一到兩個(gè)值。IIF是IF-THEN-ELSE語(yǔ)句的緊湊表達(dá)。語(yǔ)法如下:
IIF(expression, if_true, if_false)

通過(guò)該函數(shù),可以建立非常復(fù)雜的過(guò)濾字符串。例如,假定你從SQL Server的Northwind數(shù)據(jù)庫(kù)中取得Employees表,下列表達(dá)式可以選出那些employeeID小于6且lastname為偶數(shù)個(gè)字符和employeeID大于6且lastname為奇數(shù)個(gè)字符的員工。
IIF(employeeID<6, Len(lastname) %2 =0, Len(lastname) %2 >0)

下圖顯示了結(jié)果(樣品應(yīng)用程序會(huì)在稍后討論)

圖三 對(duì)Northwind中的表進(jìn)行過(guò)濾
例子程序是一個(gè)Windows® Form應(yīng)用程序,其中使用了兩個(gè)datagrid
控件來(lái)實(shí)現(xiàn)master/detail結(jié)構(gòu)。一個(gè)grid在載入時(shí)生成,即在SQL
Server data adapter完成數(shù)據(jù)讀取工作之后。請(qǐng)注意,data
adapter是Beta 2中引入的,在Beta 1中相應(yīng)的是SQLDataSetCommand類(lèi)。


預(yù)排視圖
在上面的舉例中,datagrid必須負(fù)責(zé)預(yù)排視圖中的數(shù)據(jù)行,以便刷新用戶界面。這個(gè)自動(dòng)機(jī)制是.NET
數(shù)據(jù)綁定的產(chǎn)物。Datagrid是通過(guò)DataSource屬性來(lái)獲取數(shù)據(jù)的數(shù)據(jù)綁定控件。DataView是一個(gè)可數(shù)據(jù)綁定的類(lèi),可構(gòu)建DataSource屬性的內(nèi)容。

如果你想使用datagrid之外的另一個(gè)控件,應(yīng)該怎么辦呢?又如果你不想使用自動(dòng)數(shù)據(jù)綁定呢?應(yīng)該怎樣預(yù)排視圖中所選的數(shù)據(jù)行呢?

DataView的Table屬性指向相應(yīng)的數(shù)據(jù)表,但DataTable并不保存過(guò)濾信息。所以,預(yù)排表中的數(shù)據(jù)注定是不可行的。雖然DataTableDataView是緊密相聯(lián)的,但它們各自保持獨(dú)立,并執(zhí)行獨(dú)立的功能。

以下Visual Basic .NET代碼段顯示了如何遍歷視圖中所有的數(shù)據(jù)行,并加入到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屬性存儲(chǔ)了視圖中數(shù)據(jù)行數(shù),以便在For..Next循環(huán)中使用。
要訪問(wèn)視圖中某一行,可以使用DataRowView類(lèi)。DataRowView可表示DataRow的視圖,就像DataView表達(dá)DataTable定制的視圖一樣。
總的來(lái)說(shuō),DataRow最多有四種狀態(tài):default,original,current和proposed。這些狀態(tài)由DataRowVersion枚舉類(lèi)型設(shè)置,由RowVersion屬性表達(dá)。
DataRow的視圖只能是其中某一種狀態(tài)。
數(shù)據(jù)行的默認(rèn)(default)版本只有當(dāng)其列在構(gòu)造時(shí)設(shè)定了默認(rèn)值時(shí)才有。而初始(original)版本是指在最后一次調(diào)用表的AcceptChanges后,從數(shù)劇源中得到數(shù)據(jù)行或快照。當(dāng)前(Current)版本是指當(dāng)前的數(shù)據(jù)行,包括所有當(dāng)時(shí)發(fā)生的更新。Proposed狀態(tài)只存在于調(diào)用BeginEdit和EndEdit的編輯過(guò)程中。
可以通過(guò)訪問(wèn)DataRow相同的語(yǔ)法訪問(wèn)DataRowView。這里最重要的屬性叫Item。

排序和其他便捷的特性
DataView支持Sort屬性,可以用來(lái)對(duì)視圖中的內(nèi)容排序。Sort由用逗號(hào)分隔的列名表達(dá)式進(jìn)行排序。通過(guò)在任何列名后加ASC或者DESC限定詞,可以使得字段按照上升或者下降的順序排列。如果沒(méi)有方向限定詞,默認(rèn)順序?yàn)锳SC。
DataView是內(nèi)存中的對(duì)象,所以排序在本地進(jìn)行,無(wú)需調(diào)用數(shù)據(jù)庫(kù)服務(wù)器。
RowStateFilter是DataView另一有趣的屬性。它可以用任何預(yù)定義的標(biāo)準(zhǔn)來(lái)過(guò)濾DataTable中的內(nèi)容。下表中是DataViewRowState枚舉類(lèi)型的所有取值:
CurrentRows包括所有未更新的、新的和修改的數(shù)據(jù)行
Deleted所有自上次調(diào)用AcceptChanges后刪除的數(shù)據(jù)行
ModifiedCurrent所有自上次調(diào)用AcceptChanges后修改過(guò)的數(shù)據(jù)行
ModifiedOriginal所有自上次調(diào)用AcceptChanges后original版本的數(shù)據(jù)行
New所有自上次調(diào)用AcceptChanges后新添加的行
OriginalRows返回初始數(shù)據(jù)行,包含unchanged和deleted 的
Unchanged所有未更新的數(shù)據(jù)行

如果要操作非連接的數(shù)據(jù),所有更新都在對(duì)DataTable調(diào)用AcceptChanges后生效。對(duì)單一行的更新在調(diào)用DataRowAcceptChanges后生效。類(lèi)似的,這些更新可以通過(guò)調(diào)用DataTableDataRow對(duì)象的RejectChanges來(lái)取消。
DataView對(duì)象還有一些屬性,如AllowEdit,AllowDeleteAllowNew,用來(lái)得到或設(shè)定是否允許更新的值。它們的默認(rèn)值設(shè)為T(mén)rue,允許任何種類(lèi)的更新。如果在標(biāo)志設(shè)為False時(shí),你想要完成相應(yīng)的更新操作,會(huì)有一個(gè)運(yùn)行時(shí)錯(cuò)誤發(fā)生。

DataViewManager類(lèi)
DataTable對(duì)象的DefaultView屬性用來(lái)返回一個(gè)DataView對(duì)象,作為數(shù)據(jù)表中內(nèi)容的默認(rèn)視圖。它按照自然順序讀取數(shù)據(jù)并顯示表中所有的行,而不使用任何過(guò)濾。
theMasterGrid.DataSource = m_ds.Tables("Employees").DefaultView

如果需要數(shù)據(jù)特定的視圖,你可以進(jìn)行排序并/或?qū)efaultView對(duì)象直接進(jìn)行過(guò)濾。
m_ds.Tables("Employees").DefaultView.Sort = "lastname"
theMasterGrid.DataSource = m_ds.Tables("Employees").DefaultView
DataViewManager類(lèi)是用來(lái)存儲(chǔ)DataSet中所有表的視圖設(shè)置。
可以通過(guò)傳遞一個(gè)合法的非空的DataSet給類(lèi)的構(gòu)造函數(shù)來(lái)創(chuàng)建DataViewManager
Dim dvm As DataViewManager
dvm = New DataViewManager(m_ds)

也可以通過(guò)DataSet對(duì)象的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對(duì)象的集合。

Dim dvs As DataViewSetting
dvs = dvm.DataViewSettings("Employees")
dvs.Sort = "lastname"

DataViewSetting對(duì)象包含了表視圖的參數(shù)信息。當(dāng)將數(shù)據(jù)綁定到對(duì)數(shù)據(jù)敏感的控件時(shí),使用DataViewManager而不是DataSet或DataTable可以保留你的視圖設(shè)置(過(guò)濾和排序字段)

theMasterGrid.DataSource = dvm
theMasterGrid.DataMember = "Employees"

在這里,視圖按照DataViewSetting中對(duì)Employees表指定的自動(dòng)進(jìn)行排序和過(guò)濾。換而言之,DataViewSetting類(lèi)是對(duì)特定表的視圖的一種緩存。

下一步
上述例子程序用filter實(shí)現(xiàn)了master/detail結(jié)構(gòu)。如果使用.NET中特有的數(shù)據(jù)綁定控件(如datagrid),能夠更好的達(dá)到這個(gè)目的。在以后的專(zhuān)欄中,我將論述內(nèi)存中的數(shù)據(jù)關(guān)系,以及它們是如何影響master/detail結(jié)構(gòu)的設(shè)計(jì)的。
對(duì)話:你是否需要控件或組件?

在.NET中有很多術(shù)語(yǔ)經(jīng)?梢蕴娲褂。這里特別指出的是:類(lèi),組件,對(duì)象和控件。在此,我提供了一張表,來(lái)表述每個(gè)術(shù)語(yǔ)的恰當(dāng)?shù)暮x。我們經(jīng)常將它們當(dāng)成同義詞。


需要牢記在心的是整個(gè).NET架構(gòu)是由類(lèi)組成的。所以你從中得到的任何東西,首先,是一個(gè)類(lèi)。在.NET環(huán)境中,控件和組件不是同一種類(lèi)。至于對(duì)象,可以認(rèn)為是運(yùn)行著的.NET類(lèi)的實(shí)例。
組件是一個(gè)特殊的類(lèi),它實(shí)現(xiàn)了Icomponent接口或派生于實(shí)現(xiàn)了Icomponent接口的類(lèi)。
控件是提供了用戶界面功能的組件。在.NET架構(gòu)中,可以找到兩類(lèi)控件:客戶端的Windows Forms 控件和ASP.NET server 控件。
Icomponent接口包含在Idisposable接口中,并提供了一種確定的方法清除資源。
Public Interface IComponent
Inherits IDisposable

這種釋放資源的方法和標(biāo)準(zhǔn)的.NET垃圾收集器可以二者選一。通過(guò)實(shí)現(xiàn)Idisposable,你定義了一個(gè)Dispose方法。這樣通過(guò)編程,你可以顯式的釋放對(duì)象而無(wú)須等待垃圾收集器來(lái)處理。
.NET組件知道怎樣在應(yīng)用程序域(application domain)中如何串聯(lián)。這有兩種方法:通過(guò)引用或通過(guò)值,基本功能分別內(nèi)建于MarshalByRefComponet和MarshalByValueComponent類(lèi)中。.NET component類(lèi),事實(shí)上,實(shí)現(xiàn)了Idisposable,但直接或間接繼承了上述兩個(gè)類(lèi)中的一個(gè)。
應(yīng)用程序域是一種輕量級(jí)進(jìn)程。通過(guò)引用來(lái)列集對(duì)象意味著proxy/stub實(shí)體對(duì)會(huì)被創(chuàng)建并處理遠(yuǎn)程調(diào)用。而通過(guò)值則意味著該對(duì)象的序列化的拷貝傳遞越過(guò)域的邊界。
控件是更特殊化的對(duì)象,它還提供了用戶界面元素。當(dāng)然,一個(gè)控件總是一個(gè)component,但反之不一定成立。