ADO.NET 2.0 - 如何建立一個 DataView

摘要:ADO.NET 2.0 - 如何建立一個 DataView

借助於DataView,我們便能夠替儲存於DataTable中的資料建立不同的檢視。比方說,透過DataView,您可以使用不同的排序順序檢視DataTable中的資料,亦或是根據資料列狀態或篩選運算式來篩選DataTable中的資料。重要的是,當我們需要替DataTable中的資料建立不同的檢視而且需要將這些資料繫結至表單上的控制項時,更需要使用DataView來完成之。 

DataView提供的是一種動態的資料檢視,也就是說,其內容、排序順序、與成員會反應出來源DataTable中的任何變更。顯然DataViewDataTable.Select方法有很大的不同,Select方法會根據特定的篩選運算式或排序順序從資料表中傳回一個DataRow物件陣列,而且其成員與排序順序是靜態不變的。正由於DataView的動態反應特性,因此非常適合用於資料繫結應用程式。 

DataView提供您單一資料集合的動態檢視,您可以對資料集合套用不同的排序順序和篩選條件,此點與SQL Server資料庫所提供的檢視表(View)有點類似。然而,DataView與資料庫檢視表仍然有相當大的差異,因為DataView無法當成資料表來使用,也無法提供連結資料表的檢視。此外,您不僅不能排除來源資料表中的欄位,也不能額外加入來源資料表中不存在的欄位(例如:運算式欄位)。 

您可以採用兩種方式來建立一個DataView。第一種方式是使用DataView建構函式,第二種方式則是建立DataTableDefaultView屬性的一個參考。本文將詳細探討如何使用這兩種方式來建立DataView 

使用DataView建構函式 

DataView建構函式共提供下表所示的三個多載版本。 

DataView建構函式的多載版本

DataView()

DataView(ByVal table As DataTable)

DataView(ByVal table As DataTable, _
        ByVal RowFilter As String, _
        ByVal Sort As String, _
        ByVal RowState As DataViewRowState)

DataView建構函式的第一個版本表示不使用任何參數來初始化DataView類別的新執行個體。請注意,如果您採用此版本來建立DataView,必須在建立DataView物件後先設定Table屬性以便決定其來源DataTable,然後才能繼續設定其他屬性(RowFilterSort等等)。 

以下的程式碼示範如何使用第一個版本的DataView建構函式來建立DataView物件,以便篩選和排序資料集內之「章立民工作室」資料表的資料列,並將DataGridView控制項繫結至此DataView。我們發現,DataGrid控制項只會顯示出女性,並且資料會依姓名的筆畫順序由多至少排列。以下是本範例的程式碼: 

SqlDataAdapter1.Fill(Ds章立民工作室, "章立民工作室")
'
建立 DataView 物件
Dim dv As DataView = New DataView

'
由於採用沒有任何參數的 DataView 建構函式來建立 DataView 物件,
'
因此必須先設定 Table 屬性以便決定其來源 DataTable
dv.Table = Ds章立民工作室.章立民工作室

'
設定排序順序以便依姓名的筆畫順序由多至少排列
dv.Sort = "姓名 DESC"

' 設定篩選條件以便只顯示出女性
dv.RowFilter = "性別 = ''"

' DataGridView 控制項繫結至 DataView
DataGridView1.DataSource = dv
 

DataView建構函式的第二個版本表示使用指定的DataTable來初始化DataView 類別的新執行個體。 

以下的程式碼它示範如何使用第二個版本的DataView建構函式來建立DataView物件,以便篩選和排序資料集內之「章立民工作室」資料表的資料列,並將DataGridView控制項繫結至此DataView。我們發現,DataGridView控制項只會顯示出「資訊部」的員工資料,並且資料會依照目前薪資由高至低排列: 

SqlDataAdapter1.Fill(Ds章立民工作室, "章立民工作室")

'
建立 DataView 物件
Dim dv As DataView = New DataView(Ds章立民工作室.章立民工作室)

'
設定排序順序以便依目前薪資由高至低排列
dv.Sort = "目前薪資 DESC"

' 設定篩選條件以便只顯示出「資訊部」的員工資料
dv.RowFilter = "部門 = '資訊部'"

' DataGridView 控制項繫結至 DataView
Me.DataGridView1.DataSource = dv
 

DataView建構函式的第三個版本表示使用指定的DataTableRowFilterSortDataViewRowState,來初始化DataView類別的新執行個體。 

以下的程式碼它示範如何使用第三個版本的DataView建構函式來建立DataView物件,以便篩選和排序資料集內之「章立民工作室」資料表的資料列,並將DataGridView控制項繫結至此DataView。我們發現,DataGridView控制項只會顯示出目前薪資大於49000元的員工資料,並且資料會依照目前薪資由高至低排列: 

SqlDataAdapter1.Fill(Ds章立民工作室, "章立民工作室")

'
建立 DataView 物件
Dim dv As DataView = New DataView( _
                             Ds
章立民工作室.章立民工作室, _
                             "
目前薪資> 49000", _
                             "
目前薪資 DESC", _
                             DataViewRowState.CurrentRows)
'
DataGridView 控制項繫結至 DataView

Me.DataGridView1.DataSource = dv 

在此要特別提醒大家,當DataView被建立時,以及當SortRowFilterRowStateFilter屬性有任何一個被修改時,都會重新建立DataView的索引。此意味著,如果您希望享有最佳的效能,應該在建立DataView時,直接於建構函式中指定排序順序或篩選條件。如果您於建立DataView時並未直接於建構函式中指定排序順序或篩選條件,而是於建立DataView物件之後再設定其SortRowFilterRowStateFilter屬性,將會導致DataView的索引被重新建立,而使得索引至少被建立兩次。 

使用DataTableDefaultView屬性

DataTableDefaultView屬性會傳回一個以此DataTable作為來源資料表的DataView物件,您便讓您去排序、篩選、與搜尋DataTable中的資料列。如果您所建立的DataView要顯示出DataTable中的所有資料列並依照自然順序來排列,則使用DataTableDefaultView屬性來建立DataView將是非常直接且便利的方式。 

以執行畫面如圖表1所示的程式而言,它使用DataTableDefaultView屬性來建立DataView,以便一開始能夠於DataGrid控制項中顯示出來源資料表的所有資料列,並讓使用者在執行階段透過DataView來動態篩選資料。茲將程式碼列示如下: 

' DataView 物件的類別層級宣告
Private dv As DataView

Private Sub DemoForm5_Load(ByVal sender As System.Object, _
  ByVal e As System.EventArgs) Handles MyBase.Load
  FillComboBoxDepartment()

  SqlDataAdapter1.Fill(Ds
章立民工作室, "章立民工作室")

  '
建立 DataView
  dv = Ds章立民工作室.章立民工作室.DefaultView

  '
顯示 DataView 中的資料列數目
  txtRowCount.Text = dv.Count.ToString

  '
DataGridView 控制項繫結至 DataView
  Me.DataGridView1.DataSource = dv
End Sub

Private
Sub FillComboBoxDepartment()
  '
建立資料命令物件(亦即 SqlCommand 物件)
  Dim foxCMD As New SqlCommand
  foxCMD.Connection = SqlConnection1
  foxCMD.CommandText = "SELECT DISTINCT
部門 FROM dbo.章立民工作室"

  ' 開啟連接
  SqlConnection1.Open()

  Using myReader As SqlDataReader = foxCMD.ExecuteReader()
     If myReader.HasRows Then
          While myReader.Read()
              ComboBoxDepartment.Items.Add(myReader.GetSqlString(0))
          End While
      End If
  End Using
  ComboBoxDepartment.SelectedIndex = 0
End Sub

Private
Sub btnFilter_Click(ByVal sender As System.Object, _
  ByVal e As System.EventArgs) Handles btnFilter.Click
  dv.RowFilter = "
部門= '" & _
    ComboBoxDepartment.SelectedItem.ToString() & "'"
  ' 顯示 DataView 中的資料列數目
  txtRowCount.Text = dv.Count.ToString
End Sub

圖表 1