Visual Basic 的数据库编程[3]

[入库:2005年8月18日] [更新:2007年3月24日]

本文简介:选择自 shawls 的 blog

  动态集类型的 recordset 对象可以是本地的表,也可以是返回的行查询结果。它实际上是对一个或者几个表中的记录的一系列引用。可用动态集从多个表中提取和更新数据,其中包括链接的其它数据库中的表。动态集类型具有一种与众不同的特点:不同数据库的可更新联接。利用这种特性,可以对不同类型的数据库中的表进行可更新的联接查询。动态集和它的基本表可以互相更新。如果动态集中的记录发生改变,同样的变化也将在基本表中反映出来。在打开动态集的时候,如果其他的用户修改了基本表,那么动态集中也将反映出被修改过的记录。动态集类型是最灵活的recordset 类型,也是功能最强的。不过,它的搜索速度与其它操作的速度不及表类型的 recordset。
  快照类型的 recordset 对象包含的数据是固定的,它反映了在产生快照的一瞬间数据库的状态。从 microsoft jet 数据源得到的快照是不可更新的,从开放数据库互连 (odbc) 数据源得到的某些快照是可以更新的,这取决于数据库系统本身的能力。与动态集类型和表类型的 recordset 对象相比,快照的处理开销较少。因此,它执行查询和返回数据的速度更快,特别是在使用 odbc 数据源时。快照类型保存了表中所有记录的完整复本,因此,如 果记录的个数很多,快照的性能将比动态集慢得多。为了确定快照与动态集哪一个更快,可以先以动态集方式打开记录集,然后再以快照方式打开它。
  具体使用什么记录集,取决于需要完成的任务:是要更改数据呢,还是简单地查看数据。例如,如果必须对数据进行排序或者使用索引,可以使用表。因为表类型的 recordset 对象是做了索引的,它定位数据的速度是最快的。如果希望能够对查询选定的一系列记录进行更新,可以使用动态集。如果在特殊的情况下不能使用表类型的记录集,或者只须对记录进行扫描,那么使用快照类型可能会快一些。
  一般来说,尽可能地使用表类型的 recordset 对象,它的性能通常总是最好的。
  为选择特定的 recordset 类型,把 data 控件的recordsettype属性设成:

recordset记录集属性
bof属性 当记录集记录指针指向第一条记录时返回true
eof属性 当记录集记录指针指向最后一条记录时返回true
abslouteposition属性 返回当前记录集记录指针,第一条记录为0,是只读属性
bookmark属性 string类型,返回或设置当前记录集记录指针的书签,是可读写属性。每一条记录都有自己唯一的书签,它与记录在记录集中的顺序无关。将bookmark属性存放到变量中,后面可以通过将该变量赋值给bookmark属性,并返回到这个记录。
注意:程序中使用bookmark属性重定位记录指针,而不能使用abslouteposition
nomatch属性 当我们使用find方法查询时如果未找到则返回true。常与bookmark属性同时使用。
例如:查找[name]字段中第一个姓李的人

dim s as string
with xxxx.recordset
 s = .bookmark
   .findfirst "[name] like ’李*’"
 if .nomatch then
  msgbox "数据未找到“
  .bookmark = s
 end if
end with

记录集方法
addnew方法 向记录集增加一条新记录
delete方法 从记录集中将当前记录删除。在删除后常使用movenext方法移动指针。
例如:

with xxxx.recordset
     .delete
     .movenext
 if .eof then .movelast
end with

movexxxx方法
movefirst 将记录集指针移动到第一条记录上
movelast 将记录集指针移动到最后一条记录上
moveprevious 将记录集指针移动到前一条记录上
movenext 将记录集指针移动到下一条记录上
findxxxx方法
findfirst在记录集中查询符合条件的第一条记录
findlast 在记录集中查询符合条件的最后一条记录
findprevious 在记录集中查询符合条件的前一条记录
findnext 在记录集中查询符合条件的下一条记录
  好了,有了这么充分的知识了,编写两个按钮命令简直是小菜一碟,先来试一下,添一个“增加”命令按钮吧。

private sub command1_click()
 data1.recordset.addnew
end sub

  哇!怎么这么简单,再看一下“删除”命令按钮

private sub command2_click()
 data1.recordset.delete
 data1.recordset.addnew
end sub

  就这样行了吗?运行程序吧,ok!一切正常,迫不及待地输入一条记录,点击“增加”按钮,怎么?出问题了!因为你只有在进行了addnew方法后才可以输入数据,好吧,在窗口的初始化时就增加一条新记录吧。

private sub form_initialize()
data1.recordset.addnew
end sub
  输入完了数据,我们打算退出程序,很自然的我们执行关闭窗口操作,就顺利地结束了输入工作。真的很顺利吗?打开数据库,看看数据库中的数据,我们发现刚才输入的最后一条记录没有存入数据库中。这个很好解释,每当我们调用addnew方法时,它就将输入的记录存入数据库中,而当我们关闭窗口时,刚输入的记录并没有保存到数据库中,那么在关闭窗口之前对data控件进行一次刷新就可以将数据存入数据库中了。

private sub form_queryunload(cancel as inte ger, unloadmode as integer)
 data1.refresh
end sub

  到了这里,我们似乎可以稍稍轻松了一点,这个窗口的功能差不多完成了。但是我不得不给你提出一个忠告:在数据库系统中,应尽量将错误在应用级上处理。这句话看起来似乎有点抽象,实际上用在这个程序中就简单多了。在表register中,我们将出生日期定义为date/time类型,如果在程序运行时,在该字段对应的文本框中输入的不是date/time格式,在向数据库提交数据时会出现什么情况呢?数据库会向用户报告错误信息。然而这样对应用程序并不好,这样的错误应该由用户程序处理,而不是交给数据库去处理,所以在数据提交之前就应该检查该字段的输入是否合法。

private sub text3_lostfocus()
 if isdate(text3.text) or text3.text = "" then ’检查是否输入合法数据
exit sub
end if
msgbox ("输入错误,请输入你出生的年月日!")
,将选取不合法的数据,以便重新输入,并使控制焦点不动
text3.setfocus
text3.selstart = 0
text3.sellength = len(text3.text)
end sub

  上面虽是应用程序处理错误的一个小例子,可是这种在应用级处理错误的思想是十分重要的。

五、寻寻觅觅
  在数据库管理系统中,输入和查询就象两个孪生姐妹不可或缺,下面将介绍如何创建查询窗口。查询窗口的设计分为两部分:查询结果和查询条件。查询结果是指用户所需要的数据,它包括根据查询条件查询出来的记录,但并非表中每个字段里的数据都需要提供给用户。比如在我们这个例子中,登记表中的登记号的值是用户不感兴趣的,所以在查询结果中,我们不希望显示regid字段的值。查询条件是用户提出的查询要求。比如在我们这个系统中,可以有姓名条件,当用户想知道某个人的具体情况,他可以输入此人的姓名,就查询出此人各方面的情况;也可以有年龄条件,当用户输入某个年龄段,就会查询出处于这个年龄段的所有人的信息。到底采用哪些查询条件,这需要开发者根据用户和系统的要求进行设计,其具体实现过程大都大同小异。为了节省篇幅,我们就仅以年龄为条件进行查询。查询窗口运行情况如图9。
在这个程序中我们使用了一个控件dbgrid,这个控件用来显示查询结果,选中vb的“工程”菜单下的“部件……”项,在控件标签中,选中“microsoft data bound grid control 5.0”即可,在工具箱中就会出现dbgrid控件的小图标。其使用和其它控件一样。
  为了和数据库连接,data控件是不可少的,回忆一下,该怎样设置它的属性,ok!同输入窗口一样,在connect属性中,选中“access”项。在databasename属性中,输入“c:\temp\登记.mdb”。在recordsource属性中,选中......嘿,嘿,这里稍微有点不同,如果按输入窗口那样的设置,查询结果中就会包含登记号字段了。在此属性中我们应该输入sql语句:
  select name as 姓名,sex as 性别,hometown as 籍贯,age as 年龄,birthday as 生日,company as 单位,address as 地址,zip as 邮编,telephone as 电话,fax as 传真 from register。别着急,尽管这条语句有点长,实际上却比较简单。这条语句的语法是:
  select 字段名,字段名,……from 表名 where 条件;
  对照语法,我们可以看出输入的sql语句的含义:从表register中查询姓名,性别,籍费,……字段的值。只要在字段列表中不选中登记号字段,在查询结果中,就不会显示登记号的值了。如果你够细心的话就会注意到我们所写的sql语句中在字段列表中并不仅仅输入字段名,在其后面还增加了as……项,如“name as 姓名”,这是为name字段取一个别名“姓名”,以便在dbgrid控件中显示字段名时,就会显示“姓名”而不是“name”。
  完成了data控件属性的设置就可以将dbgrid控件捆绑到data控件上,其方法同输入窗口。对了,将“datasource”属性设置为“data1”即可。现在不妨运行一下程序,真令人兴奋,dbgrid显示出表中所有的信息。可是怎样显示符合条件的数据呢?再看一看上面的sql语法,where段后可以输入查询条件,比如:需要年龄在20到30岁之间的人员信息,其语句为:select name,... from register where age>20 and age<30;
  我们只要根据用户输入的条件构成新的sql语句,并利用data控件的refrensh方法刷新数据库,就可以完成条件查询了。
  整个程序十分简单,当用户在文本框中输入年龄段后,点按“查询”命令,就会显示符合条件的查询结果。程序代码如下:

option explicit

本文关键:Visual Basic,数据库.编程
 

本站最佳浏览方式为 分辨率 1024x768 IE 6.0(或更高版本的 IE浏览器)

go top