2 获取数据
连接数据库之后,就可以从数据库中提取数据了。在示例程序中,点击“从数据库获取数据”按钮,可以看到状态栏上的提示信息——“数据已取回,共324条”。
提取数据也是个较复杂的过程,需要有好几个对象通力协作才能完成。
(1)定义一个OleDbDataAdapter对象,这个对象负责从数据库中提取数据:
Private da As OleDbDataAdapter = Nothing
(2)定义一个OleDbCommand对象,此对象负责向数据库发送SQL命令:
Private comm As OleDbCommand = Nothing
(3)定义一个DataTable对象,此对象用于存放数据,相当于数据库中的表。
Private dt As DataTable
这三个对象加上前面介绍的OleDbConnection对象,就可以实现数据的提取。这是由Sub过程OnGetData()实现的:
1 Private Sub OnGetData()
2 If da Is Nothing Then
3 da = New OleDbDataAdapter
4 End If
5
6 If comm Is Nothing Then
7 comm = New OleDbCommand
8 comm.Connection = conn '指定连接对象
9 comm.CommandType = CommandType.Text '指明使用SQL命令
10 comm.CommandText = "Select * from Client" '指定SQL命令内容
11 End If
12 '将OleDbCommand对象与OleDbDataAdapter对连接起来。
13 da.SelectCommand = comm
14
15 If dt Is Nothing Then
16 '创建一个DataTable对象,存放从数据库中取出来的数据
17 dt = New DataTable
18 End If
19
20 '填充数据
21 da.Fill(dt)
22
23 '激活“在网格中显示记录”按钮
24 Me.btnShowDataInGrid.Enabled = True
25
26 '显示信息
27 Me.StatusBar1.Text = "数据已取回,共" & dt.Rows.Count & "条"
28
29 End Sub
注意第8句,第13句和第21句,这三条语句实现了对象之间的合作。
第8句的含义是OleDbCommand对象采用OleDbConnection对象连接数据库,通过这一连接向数据库发送SQL命令。
第13句表明OleDbAdapter对象使用OleDbCommand对象向数据库发送提取数据的“Select …”命令。
第21句表明OleDbAdapter对象将取回的数据放到DataTable对象中。
这四个对象之间的关系可以用以下图进行说明:
图 5 11 ADO.NET对象的合作关系
图 5 11 中,程序向数据库发送的各种命令(比如连接数据库,执行SQL命令等)被称为“命令流”,可以看到命令主要由OleDbAdapter负责确定,它委托OleDbCommand对象执行此特定的命令,而OleDbCommand需要通过OleDbConnection对象与数据库连接上之后,再把此命令传给数据库本身,由数据库管理系统(DBMS)执行此命令。
如果被执行的命令是一条提取数据的命令,则数据库会把需要的数据准备好,通过OleDbConnection对象建立的连接传给OleDbAdapter,由它负责把数据“塞入”DataTable对象中。被传送的数据在图 5 11中被称为“数据流”,以虚线表示。传送数据的终点站是DataTable中,而OleDbAdapter可以看成是一个数据的中转站,其Fill方法将收到的数据再转给DataTable对象。一旦数据从数据库“搬家”到DataTable之后,这些数据就自立门户,与数据库“老家”中的数据没有任何联系了。
提示:
此处只是简单介绍了一种最常见的数据流动情况,更完整详细的介绍请参见第七章。
3 显示记录
数据到达DataTable之后,并不能自动地显示在窗体上,这是因为DataTable是个非可视的对象。
在示例程序中点击“在网格中显示记录”,运行结果如图 5 12所示。
将DataTable中的数据显示在可视化控件DataGrid中是通过以下代码实现的:
DataGrid1.DataSource = dt
通过设定DataGrid对象的DataSource属性为DataTable,就在DataTable与DataGrid中建立了一种关联,DataTable中数据的任何变化都会同步地在DataGrid中显示,同样的,用户也可以通过DataGrid来直接修改数据,他所做的所有修改都会被保存到DataTable中。
有个专门的术语用于描述DataTable与DataGrid之间建立的这种关联,这就是“数据绑定(DataBinding)”。
数据绑定是个非常重要的概念,将在7.5节中对这一概念的真正含义作完整的解释。
4 编辑与修改记录
任何一个数据库应用程序,如果只能显示数据而不能增删改数据,都是用途有限的。.NET所提供的DataGrid是一个功能强大的数据绑定控件,直接支持对记录的增加、删除与修改 如图 5 13 所示,在网格中点击某个单元格,就可以直接修改这一字段的内容,并在当前行左端显示一支小铅笔
图标。
直接点击最后一行的任意单元格,输入新值,就可以新增记录,如图 5 14所示:
点击最左边的一列,可以选中一行,再按键盘上的“DEL”键,可以删除记录。
虽然通过DataGrid可以对记录进行增、删、改,但如果这时直接关闭程序,当再次运行程序时,会发现所做的修改都没有被保存。其原因何在?
修改不能保存的最根本原因是数据一旦从数据库中取出,这些数据独立于数据库了,换言之,取出的数据存放在内存中,对它的修改不会导致数据库的内容随之改变。这种数据独立性给我们带来了很大的灵活性。
注意:
VB6中使用DataGrid显示数据库中数据时,其增删改会直接保存到数据库中,即在任何时候,在窗体上看到的数据就是数据库中的数据(除非采用了“断开”方式)。VB6所使用的数据库技术是ADO,而VB.NET使用的是ADO.NET,两者有着根本的不同。VB6程序员要注意这个重要的区别。
但确实需要将修改后的数据永久保存,为此,可以使用OLEDBDataAdapter对象的Update()方法实现此功能。
试一试:
在示例程序中,给“保存”按钮的单击事件加入下面一句:
'将更改保存到数据库中
da.Update(dt)
然后运行程序,随意修改、删除或增加几条记录,单击“保存”按钮。观察运行情况。
可以看到,直接调用Update()方法时,会引发一个异常,如 图 5 15 所示:
图 5 16 更新异常
其原因何在?
数据的提取与保存都是由OLEDBDataAdapter负责的,它通过四个OLEDBCommand对象类型的属性来完成四种典型的数据处理任务:
提取记录:SelectCommand
新增记录:InsertCommand
删除记录:DeleteCommand
修改记录:UpdateCommand
当我们从数据库中提取记录时,已向OLEDBDataAdapter传入了一个OLEDBCommand对象,其执行的SQL命令是:
Select * from Client
但我们并没有给其它三个操作提供有效的OLEDBCommand对象,所以,在示例程序中OLEDBDataAdapter无法保存数据。
OLEDBCommand对象是专用于执行SQL命令的,InsertCommand执行SQL中的Insert命令,DeleteCommand执行SQL中的Delete命令, UpdateCommand执行SQL中的Update命令。
提示:
在第六章中,将介绍SQL的常用命令,学会编写SQL语句完成许多数据处理工作。
可以手动地创建OLEDBCommand对象并设置其SQL命令,然后把它传给OLEDBDataAdapter对象,但这是一个麻烦且易出错的工作,.NET提供了一个OleDbCommandBuilder类来完成此工作。参见以下代码:
Private cb As OleDbCommandBuilder
'保存数据
Private Sub OnSaveData()
If cb Is Nothing Then
cb = New OleDbCommandBuilder
cb.DataAdapter = da
'设定更新命令
da.UpdateCommand = cb.GetUpdateCommand
'设定插入命令
da.InsertCommand = cb.GetInsertCommand
'设定删除命令
da.DeleteCommand = cb.GetDeleteCommand
End If
'将更改保存到数据库中
da.Update(dt)
End Sub
请注意上面的代码中没有看到设置OLEDBDataAdapter对象的SelectCommand属性的语句,这是因为这一属性在提取数据时就已经设置了。
在单击“保存”按钮时调用上述Sub过程,数据便可以正确地保存到数据库中。
图 5 17 将数据保存到数据库中
至此,一个功能基本完备的数据库应用程序已开发完成。
看上去,一切并不复杂。但在简单的背后,是由复杂的技术做支撑的,一切并不“简单”。
5 小结
通过开发一个小的数据库应用程序,读者应该对.NET中如何开发数据库应用程序有了一个总体的认识,但这个小程序的功能还是很有限的,比如没有查找数据的功能,而且过多地依赖于DataGrid这一功能强大的控件,它把许多技术细节给封装了起来。不管我们的这个小程序多么简陋,这都是一个很好的起点。
“盲人摸象”是一个众人熟知的成语,由于软件技术系统的庞大与复杂,自学过程就是“盲人摸象”的过程。“盲人摸象”学到的东西是片面的,局部的,是应该尽力避免出现的情况。
如果把.NET数据库编程看作一头“大象”,那么,这一章的目的就是先让读者初步地了解大象是个什么样的,有了个总体印象,然后我们再用放大镜去仔细研究“大象”的每个局部,了解清楚之后,再从总体上对大象整体进行进一步的概括。通过这种方法,就可以把“大象”研究透了,“盲人摸象”的情况就不会发生。
在后面的几章中,将详细解析数据库编程的各个方面,读者通过后几章的学习,将会深入地把握ADO.NET内部的运行机理,极大地增强开发数据库应用程序的能力。