VBGood网站全文搜索 Google

搜索VBGood全站网页(全文搜索)
首页 - 经验之谈 - VB中远程数据库的访问
发表评论(0)作者:不详, 平台:VB6.0+Win98, 阅读:10207, 日期:2000-10-26
VB中远程数据库的访问

时间:2000/10/12 16:30 作者:许蔓舒


  在VB中,用microsoft jet数据库引擎和数据访问对象dao(data access object)可以创建功能强大的客户/服务器应用程序。对远程数据库的访问是开发这类应用程序的关键环节,本文将介绍在VB中用dao通过miscrosoft jet数据库引擎访问远程数据库的方法。


  用dao访问远程数据库大体上可以通过三步来实现,即数据连接、数据处理和断开连接。下面主要介绍数据连接和数据处理的具体操作。


 


一、数据连接


  dao一般通过链接远程表的方式来进行数据连接。这样,数据虽然驻留在远程数据源上,但在本地的microsoftjet数据库中可以存储与远程数据的永久性连接,同时缓存链接的表结构信息,从而在下一次访问该表时,不用再次从服务器中检索这些结构信息,加快了连接速度。一旦链接了一个表,该链接便会保留在各会话期间,直到连接断开。链接远程表的具体操作是:


用opendatabase方法打开将要包含该链接的本地microsoft jet数据库


用createtabledef方法在该数据库中创建一个新的tabledef对象


将tabledef对象的connect属性设置为一个合法的连接字符串,标识要访问的远程数据库类型、数据文件的路径以及用户名和远程数据源密码等。


将tabledef对象的sourcetablename属性设置为远程数据库中要访问的表的名称。


添加tabledef对象到tabledefs集合中。


实现链接远程表操作的过程如下:


public sub linktable(strdb as string, strrodb as string, strcn as string, strtdf as string, _linktdfname as string)


dim linktdf as new tabledef


set dbs = opendatabase(strdb)


linktdf.name = linktdfname


100


temptable = ucase(linktdf.name)


for i = 0 to dbs.tabledefs.count - 1


if ucase(dbs.tabledefs(i).name) = temptable then


if msgbox(linktdfname + " 已 存 在, 是 否 删 除 ?", _


vbquestion + vbyesno) = vbyes then


dbs.tabledefs.delete linktdf.name


exit for


else: msgbox " 重 新 输 入 新 表 名"


linktdfname = inputbox(" 新 表 名")


goto 100


end if


end if


next i


set linktdf = dbs.createtabledef(linktdfname) ’ 链 接 远 程 表


linktdf.connect = ";database=" + strcn


linktdf.sourcetablename = strtdf


dbs.tabledefs.append linktdf


end sub


上述过程用来实现远程表的连接,它有5个参数,其中strrodb是要访问的远程数据库名(包括路径);strtdf是该数据库中的表名;strdb是要链接的本地数据库(包括路径);linktdfname是本地数据库的一个新表名,用来建立远程表的链接;strcn是指定连接信息的字符串。需要特别注意的是,除了在访问远程microsoft jet数据库时,连接字符串要以分号(;)开头外,指定连接信息的字符串都必须以所访问的远程数据库类型开头。dao可以访问的远程数据源有以下三类:


.microsoft jet数据源,如:access数据。


.iisam(可安装的索引化顺序访问方法)格式数据源,如:foxpro、paradox、dbase数据。


.odbc数据源,如:sqlserver数据、oracle数据。


例如:设网络服务器名为server,共享目录为c:\sales的foxpro3.0数据库,连接字符串应为


strcn="foxpro3.0;database=\\server\c$\sales\region1"


此外,dao通过microsoft jet数据库引擎访问远程数据时,还可以用opendatabase方法直接打开远程表。在本地数据库中并未存储与远程数据源建立连接所需要的信息。如果使用链接方式访问数据,则不必在每次会话开始时提供连接信息,从而可以提高效率。


 


二、数据处理


数据连接建立后,可以用openrecordset方法打开一个记录集,并可用dbgrid控件和data控件方便地浏览整个记录集。如果使用表类型(table-type)记录对象,则对应的是一个实际存在的数据库表,在多用户环境下,其它用户对数据的修改会立即反映到表中;如果使用动态集类型(dynaset-type)记录对象,则对应的既可以是一个表中全部记录,又可以是一个查询的结果,并且可以更新记录集中的记录;如果使用快照类型(snapshot-type)记录对象,则对应的可以是表中的全部记录,也可以是一个查询结果,但不能进行记录的增加、删除和修改操作。此外,还可以建立其它类型的记录对象,如仅向前型(forward only-type)记录对象和动态型(dynamic-type)记录对象。


下面是打开动态集记录对象并显示记录的过程:


public sub rst_display(strdb as string, strrst as string, strform as form)


set dbs = opendatabase(strdb)


set rst = dbs.openrecordset(strrst, dbopendynaset)


strform!data1.databasename = dbs.name


strform!data1.recordsource = rst.name


 


strform!data1.refresh


strform!dbgrid1.rebind


end sub


上述过程有三个参数,其中strdb用来指定本地数据库名(包括路径),linktdfname是在本地数据库中新建的链接远程表的表名,strform是网格控件和数据控件所在的窗体名。调用此过程可以基于新表建立一个动态集类型的记录对象,并可在网格中浏览各个记录。


断开连接可以通过关闭应用程序或设置连接超时来实现。注意:如果对数据库对象使用close方法,则由于在miscrosoft jet数据库引擎内部缓存了连接,实际上连接并未取消。

三、应用举例


以上介绍了用dao访问远程数据库的具体操作,下面通过一个例子说明链接远程表和建立记录集对象的方法。


首先建立一个新工程,在窗体上画5个命令按钮,1个数据控件和1个数据网格控件(dbgrid)。各对象的属性设置见表1。


表1 窗体1对象属性设置




对 象
标 题(caption)
名 称(name)




窗体
远程数据访问
form1




命令按钮1
链接远程表
cmd 链接




命令按钮2
添加
cmdadd




命令按钮3
删除
cmddel




命令按钮4
修改
cmdmodify




命令按钮5
结束
cmdend




数据控件
data1
data1




数据网格
dbgrid1
   








其中dbgrid1中的datasource属性设为data1,命令按钮2,3,4的visible属性设为false。


编写如下事件过程:


private sub cmdadd_click() ’ 添 加 记 录 子 过 程


on error goto errhandler


with rst


.addnew


for i = 0 to .fields.count - 1 ’ 遍 历 记 录 集 中 的 每 个 字 段


’ 在 输 入 框 中 输 入 各 字 段 的 数 据


.fields(i).value = inputbox


(" 输 入 记 录 信 息" & vbcr + " 字 段 名:" + .fields(i).name)


next i


.update


end with


data1.refresh


dbgrid1.rebind


errhandler: ’ 错 误 处 理


select case err


case 3022, 3421


msgbox (error + vbcr + " 输 入 无 效")


exit sub


case else


response = 0


exit sub


end select


end sub


 


private sub cmddel_click() ’ 删 除 记 录 过 程


on error goto errhandler


begintrans ’ 事 务 处 理


with data1.recordset


if .bof and .eof then exit sub


’ 如 果 没 有 记 录, 退 出 过 程


.delete ’ 删 除


if .bof and .eof then


’ 如 果 没 有 记 录, 退 出 过 程


exit sub


elseif .eof then .movelast


’ 如 果 删 除 的 是 最 后 一 条 记 录, 光 标 移 至 最 后 一 记 录


else: .movenext


’ 移 至 下 一 条 记 录


end if


end with


if msgbox(" 确 实 要 删 除 这 一 记 录 ?",


vbquestion + vbyesno) = vbyes then


committrans ’ 确 认


data1.refresh


else


rollback ’ 撤 消 改 动


data1.refresh


end if


errhandler: ’ 错 误 处 理


select case err


case 3021 ’ 无 当 前 记 录


msgbox (" 无 当 前 记 录, 请 选 择 要 删 除 的 记 录")


exit sub


case else


msgbox (error)


exit sub


end select


end sub

private sub cmdend_click()


end ’ 结 束 应 用 程 序


end sub



private sub cmdmodify_click()


dbgrid1.allowupdate = true ’ 允 许 修 改


end sub



privatev sub cmd 链 接_click()


form1.hide


form2.show


end sub



private sub dbgrid1_aftercolupdate(byval colindex as integer)


’ 数 据 修 改 后 触 发 该 事 件


on error goto err1


data1.refresh


err1:


select case err


case 0


response = 0


case else


exit sub


end select


end sub



private sub dbgrid1_beforecolupdate


(byval colindex as integer, oldvalue as variant, cancel as integer)


’ 数 据 修 改 前 触 发 该 事 件


on error goto errhandler:


begintrans


if msgbox(" 确 实 要 修 改


这 一 内 容 ?", vbquestion + vbyesno) = vbyes then


committrans


else


rollback


data1.refresh


end if


errhandler:


select case err


case 0


response = 0


case else


msgbox (error)


exit sub


end select


end sub


private sub form_load()


’ 在 窗 体 装 入 时, 网 格 中 的 数 据 不 可 添 加, 修 改


dbgrid1.allowaddnew = false


dbgrid1.allowupdate = false


end sub


private sub form_resize()


on error resume next


’ 当 窗 体 调 整 时 会 调 整 网 格


dbgrid1.height = me.scaleheight - data1.height - cmd 删 除.height - 30


end sub


在工程中添加一个窗体,在窗体上画6个标签,1个命令按钮(标题为“ 确认”,名称为cmd 确认),3个文本框和1个组合框。


在窗体的声明部份输入以下代码:


’ 声 明 窗 体 层 变 量


dim rodbs as database


dim strdb as string, strrodb as string, strcn as string, strtdf as string


dim linktdfname as string


编 写 如 下 事 件 过 程:


private sub cmd 确 认_click()



on error goto errhandler:


strdb = text2.text


’ 本 地 数 据 库 名 及 路 径


linktdfname = text3.text


’ 本 地 数 据 库 中 新 建 的 链 接 远 程 表 的 表 名


strcn = strrodb ’ 连 接 字 符 串


strtdf = combo1.text ’ 指 定 远 程 数 据 库 中 要 访 问 的 表


’ 调 用linktable 过 程


call linktable(strdb, strrodb, strcn, strtdf, linktdfname)


’ 调 用rst_display 过 程


call rst_display(strdb, linktdfname, form1)


form2.hide


form1.show


form1.caption = " 远 程 数 据:" + strcn + "-" + strtdf


’ 显 示“ 添 加”,“ 删 除”,“ 修 改” 控 件


form1.cmdadd.visible = true


form1.cmddel.visible = true


form1.cmdmodify.visible = true


errhandler:


select case err


case 0


response = 0


case else


msgbox (error + vbr + " 重 新 输 入")


exit sub


end select


end sub


private sub combo1_gotfocus()


strrodb = text1.text ’ 指 定 远 程 数 据 库 名 及 路 径


set rodbs = opendatabase(strrodb) ’ 打 开 远 程 数 据 库


’ 删 除combo1 中 的 内 容


if combo1.listcount >= 1 then


for i = combo1.listcount - 1 to 0 step -1


combo1.removeitem i


next i


end if


’ 把 数 据 库 中 的 表 名 加 到combo1 中


for i = 0 to rodbs.tabledefs.count - 1


combo1.additem rodbs.tabledefs(i).name


next i


end sub

private sub cmdend_click()


end ’ 结 束 应 用 程 序


end sub



private sub cmdmodify_click()


dbgrid1.allowupdate = true ’ 允 许 修 改


end sub



privatev sub cmd 链 接_click()


form1.hide


form2.show


end sub



private sub dbgrid1_aftercolupdate(byval colindex as integer)


’ 数 据 修 改 后 触 发 该 事 件


on error goto err1


data1.refresh


err1:


select case err


case 0


response = 0


case else


exit sub


end select


end sub



private sub dbgrid1_beforecolupdate


(byval colindex as integer, oldvalue as variant, cancel as integer)


’ 数 据 修 改 前 触 发 该 事 件


on error goto errhandler:


begintrans


if msgbox(" 确 实 要 修 改


这 一 内 容 ?", vbquestion + vbyesno) = vbyes then


committrans


else


rollback


data1.refresh


end if


errhandler:


select case err


case 0


response = 0


case else


msgbox (error)


exit sub


end select


end sub


private sub form_load()


’ 在 窗 体 装 入 时, 网 格 中 的 数 据 不 可 添 加, 修 改


dbgrid1.allowaddnew = false


dbgrid1.allowupdate = false


end sub


private sub form_resize()


on error resume next


’ 当 窗 体 调 整 时 会 调 整 网 格


dbgrid1.height = me.scaleheight - data1.height - cmd 删 除.height - 30


end sub


在工程中添加一个窗体,在窗体上画6个标签,1个命令按钮(标题为“ 确认”,名称为cmd 确认),3个文本框和1个组合框。


在窗体的声明部份输入以下代码:


’ 声 明 窗 体 层 变 量


dim rodbs as database


dim strdb as string, strrodb as string, strcn as string, strtdf as string


dim linktdfname as string


编 写 如 下 事 件 过 程:


private sub cmd 确 认_click()



on error goto errhandler:


strdb = text2.text


’ 本 地 数 据 库 名 及 路 径


linktdfname = text3.text


’ 本 地 数 据 库 中 新 建 的 链 接 远 程 表 的 表 名


strcn = strrodb ’ 连 接 字 符 串


strtdf = combo1.text ’ 指 定 远 程 数 据 库 中 要 访 问 的 表


’ 调 用linktable 过 程


call linktable(strdb, strrodb, strcn, strtdf, linktdfname)


’ 调 用rst_display 过 程


call rst_display(strdb, linktdfname, form1)


form2.hide


form1.show


form1.caption = " 远 程 数 据:" + strcn + "-" + strtdf


’ 显 示“ 添 加”,“ 删 除”,“ 修 改” 控 件


form1.cmdadd.visible = true


form1.cmddel.visible = true


form1.cmdmodify.visible = true


errhandler:


select case err


case 0


response = 0


case else


msgbox (error + vbr + " 重 新 输 入")


exit sub


end select


end sub


private sub combo1_gotfocus()


strrodb = text1.text ’ 指 定 远 程 数 据 库 名 及 路 径


set rodbs = opendatabase(strrodb) ’ 打 开 远 程 数 据 库


’ 删 除combo1 中 的 内 容


if combo1.listcount >= 1 then


for i = combo1.listcount - 1 to 0 step -1


combo1.removeitem i


next i


end if


’ 把 数 据 库 中 的 表 名 加 到combo1 中


for i = 0 to rodbs.tabledefs.count - 1


combo1.additem rodbs.tabledefs(i).name


next i


end sub