说三道四技术文摘-感悟人生的经典句子
说三道四 > 文档快照

在DELPHI程序中使用ADO对象存取ODBC数据库

HTML文档下载 WORD文档下载 PDF文档下载
在DELPHI程序中使用ADO对象存取ODBC数据库

关键词:Delphi - MIS

作为一个ASP爱好者,笔者经常在ASP页面中使用ADO对象操作ODBC数据库,觉得用ASP创建Web应用系统确实是挺方便的。虽然在编程生涯中,笔者更喜欢Borland系列产品,对微软产品有点排斥,但对ASP却是例外。某天,忽然想到,ADO对象是一个标准OLE对象,如果在Delphi应用程序中能利用ADO操作数据库,应该挺不错。尤其在用DELPHI做网络数据库应用程序时,如果所在的Web站点是WINNT站点并且支持ASP页面,那就可以用ADO对象访问ODBC数据库,而不用把那么大的BDE再上载到站点上去。这样就可充分利用DELPHI和ASP的编程优势,做出更好的ISAPI/NSAPI/CGI。经过编程和测试,在Delphi中可以成功地用ADO对象存取ODBC数据库,现将本人的使用经验写出来,与大家共享,让我们多一个访问ODBC数据库的方法。

在32位的Delphi中,可以声明一个variant变量(如AVariant),然后通过CreateOleObject创建一个OLE对象,如AVariant:=CreateOleObject ('ADODB.Connection')可以获得一个数据库连接对象的实例,然后就可以利用该对象的方法和属性来操作ODBC数据库了。

下面简单介绍一下访问ODBC数据库所用到的ADO对象及其方法和属性。

1、数据库连接对象(ADODB. Connection)

该对象用于与ODBC数据库建立连接,所有对数据库的操作均通过该连接进行。

数据库连接对象ADODB. Connection的作用象Delphi中的TDatabase对象。

建立一个连接对象的方法为(AConnection为Variant类型变量):

AConnection:=CreateOleObject('A DODB.Connection')

用于建立连接的方法为Open,使用语法为(以对象AConnection为例):

AConnection.Open( ConnectionString

UserId

Password )

三个参数均为字符串类型,其中UserId和Password为用户名称和用户密码,用于访问数据库使用时可以省略,因为在ConnectionString同样可以指定用户名称和用户密码。ConnectionString是用来说明ODBC数据源信息的字符串,其格式为:

'Provider=ProviderName; DSN=DSNName; DRIVER=driver; SERVER=server; DATABASE=database; UID=user; PWD=password'

其中:

Provider:数据提供者,默认状态下为MSDASQL,为微软OLEDB,通常省略;

DSN:要打开的数据库对应的OBDC系统数据源(DSN),是可选参数;

DRIVER:要打开的数据库所用的驱动程序名称,如Access对应Microsoft Access Driver (*.mdb),是可选参数;

SERVER:要打开的数据库所在的服务器名称,本机可用(local),是可选参数;

DATABASE:要打开的数据库名称,是可选参数;

UID:用户名称,用来访问数据库,是可选参数;

PWD:用户密码,用来访问数据库,是可选参数。

以上参数均为可选参数,但必须提供足够的信息来描述一个系统数据源。

假如已经定义了一个ODBC的系统DSN,名称为MyDsn,那么就可用以下语句建立一个数据库连接:

AConnection.Open('DSN=MyDsn');

为了防止DSN不存在或其设置被他人修改时造成应用程序运行错误,可以用ADODB.Connection 创建一个临时ODBC数据源,这样可以保证我们使用的系统DSN的参数设置是正确的。下面的语句可以创建一个临时ODBC系统DSN,对应一个ACCESS数据库,路径为C:\Inetpub\ wwwroot\ test.mdb:

AConnection.open('Driver= {Microsoft Access Driver (*.mdb)};DBQ=C:\inetpub \wwwroot\test.mdb')

建立一个ADODB.Connection后,如果不需要返回操作结果(如删除,修改,更新等操作)就可以对数据库进行正常的SQL操作了,此时应用ADODB.Connection的另外一个方法Execute,使用语法为:

AConnection.Execute( strSQL );

其中strSQL为执行操作的SQL语句,如删除操作可以为:delete from wfjcommu。用AConnection.Close关闭一个数据库连接。

2、数据集对象(ADODB. RecordSet)

如果要执行查询操作并返回查询结果,或者要更方便地操作数据表,那就需要用到数据集对象了。

数据集对象ADODB.RecordSet的作用象Delphi中的TTable或TQuery对象。

建立一个数据集对象的方法为(ARecordSet为Variant类型变量):

ARecordSet:=CreateOleObject ('ADODB.RecordSet')

从数据表取得数据的方法为Open方法,具体使用方法为:

ARecordSet.Open( strCommand

ActiveConnection

int CursorType

intLockType

intCommandType );

其中:

strCommand:字符串,为命令参数,可以是一个Table名称,可以是一个SQL语句,也可以是一个服务器上的存储过程(StoredProc)名称,具体需要后面的参数intCommandType来指定。

ActiveConnection:要使用的数据库连接,是一个ADODB. Connection对象。

intCursorType:长整数,数据集的Cursor类型,可选参数,请参见程序中注释。

intLockType:长整数,对数据表的加锁类型,可选参数,请参见程序中注释。

intCommandType:长整数,命令参数的类型,用来指明strCommand的作用,可以指定strCommand为命令(如SQL语句)或数据表(TTable)或储存过程(StoredProc),可选参数,请参见程序中注释。

如执行一个SQL查询,可以采用如下语句:

ARecordSet.Open('Select * from wfjcommu'

adOpenStatic

ad LockOptimistic

adCmdText);

其它常见属性和方法与TTable和TQuery相比较如下(具体请见ASP帮助文件):

eof

bof: eof

bof.MoveFirst

MoveLast: First

LastMovePrevious

MoveNext: Prior

NextMove: MoveByAddNew: appendUpdate: PostClose: close

Delete加Update:delete,所有对数据表的修改均须用Update使操作有效,这与Delphi不同。

Fields[FieldNo]: Fields[FieldNo]

Fields['FieldName']: FieldByName('FieldName')

3、其他常见对象(与Delphi对应的对象):

ADODB.Field: TField ADODB.Parameter: TPara ADODB.Error: EDBEngineError

ADODB.Command:无 ADODB.Property:无

下面来看一个应用例子:

procedure TForm1.Button1Click(Sender: TObject);

{*****************************************************

用ADO操作ODBC数据库

本程序中,将创建一个临时的ODBC系统数据源,指向一个MsAccess数据库,然后对其中的数据表进行显示、增加、修改、删除和查询操作

注意:请在Uses语句中包含ComObj单元

*****************************************************}

const{一些常量声明,详细请参见adovbs.inc}

{---- CommandType的常量说明 ----}

adCmdUnknown = 0008;//未知

需要系统来判断

速度慢

为缺省值

adCmdText = 0001;//命令语句如SQL语句

adCmdTable = 0002;//数据表名称

adCmdStoredProc = 0004;//存储过程名称

{---- CursorType的常量说明 ----}

adOpenForwardOnly = 0;//只能由前向后单向访问

为缺省值

adOpenKeyset = 1;//可见其他用户对数据的修改

但对其它用户的增加和删除不可见

adOpenDynamic = 2;//其他用户对数据的增加修改和删除均可见

adOpenStatic = 3;//其他用户对数据的增加修改和删除均不可见

{---- LockType的常量说明 ---}

adLockReadOnly = 1;//只读

为缺省值

adLockPessimistic = 2;//在修改时

按单个记录锁定

adLockOptimistic = 3;//在修改后更新时

按单个记录锁定

adLockBatchOptimistic = 4;//在成批更新时记录锁定

var

AConnection

ARecordSet : variant;

longintTemp : integer;

strTemp : string;

intIndex : integer;

begin

{创建一个临时的ODBC数据源

指向一个MsAccess数据库

并利用此DSN建立一个数据库连接}

AConnection := CreateOleObject('ADODB.Connection');

AConnection.Open('Driver={Microsoft Access Driver (*.mdb)};DBQ=C:\inetpub\wwwroot\test');

{建立一个数据集对象

并从数据表中提取数据}

ARecordSet := CreateOleObject('ADODB.RecordSet');

ARecordSet.open( 'wfjcommu'

AConnection

adOpenStatic

adLockOptimistic

adCmdTable );

memo1.lines.clear;

memo1.lines.add('********数据表原有的内容如下********');

{显示各个域的域名}

strTemp := '';

for intIndex := 0 to ARecordSet.Fields.count - 1 do

strTemp := strTemp + ARecordSet.Fields[intIndex].name+';';

memo1.lines.add( strTemp );

{显示各个域的内容}

while not ARecordSet.eof do

begin

strTemp := '';

for intIndex := 0 to ARecordSet.Fields.count - 1 do

strTemp := strTemp + ARecordSet.Fields[intIndex].value+';';

memo1.lines.add( strTemp );

ARecordSet.MoveNext;//移到下条,Next

end;

{增加一个记录}

ARecordSet.AddNew;//增加

Append

ARecordSet.Fields['AName'] := '1';//以FieldByName的方式存取

ARecordSet.Fields['Portable'] := '2';

ARecordSet.Fields(2) := '3';//以Fields[index]的方式存取

ARecordSet.Fields(3) := '4';

ARecordSet.Fields(4) := '5';

ARecordSet.Update;//更新,Post

ARecordSet.MoveFirst;//移到首条,First

memo1.lines.add('********增加了一条记录后的数据表的内容如下********');

{显示各个域的内容}

while not ARecordSet.eof do

begin

strTemp := '';

for intIndex := 0 to ARecordSet.Fields.count - 1 do

strTemp := strTemp + ARecordSet.Fields[intIndex].value+';';

memo1.lines.add( strTemp );

ARecordSet.MoveNext;//移到下条,Next

end;

{修改最后一条记录}

ARecordSet.MoveLast;

ARecordSet.Fields['AName'] := '11';//以FieldByName的方式存取

ARecordSet.Fields['Portable'] := '22';

ARecordSet.Fields(2) := '33';//以Fields[index]的方式存取

ARecordSet.Fields(3) := '44';

ARecordSet.Fields(4) := '55';

ARecordSet.Update;//更新,Post

ARecordSet.MoveFirst;//移到首条,First

memo1.lines.add('********修改了最后一条记录后的数据表的内容如下********');

{显示各个域的内容}

while not ARecordSet.eof do

begin

strTemp := '';

for intIndex := 0 to ARecordSet.Fields.count - 1 do

strTemp := strTemp + ARecordSet.Fields[intIndex].value+';';

memo1.lines.add( strTemp );

ARecordSet.MoveNext;//移到下条,Next

end;

{删除最后一条记录}

ARecordSet.MoveLast;//移到末条,Last

ARecordSet.delete;//删除,delete

ARecordSet.Update;//更新,在Delphi不需要

ARecordSet.MoveFirst;//移到首条,First

memo1.lines.add('********删除了最后一条记录后的数据表的内容如下********');

{显示各个域的内容}

while not ARecordSet.eof do

begin

strTemp := '';

for intIndex := 0 to ARecordSet.Fields.count - 1 do

strTemp := strTemp + ARecordSet.Fields[intIndex].value+';';

memo1.lines.add( strTemp );

ARecordSet.MoveNext;//移到下条,Next

end;

ARecordSet.Close;{关闭数据集}

{用SQL语句进行查询

查询姓名为“张三”的记录}

{注意,在SQL语句中,字符串应该用单引号包括起来}

ARecordSet.open( 'select * from wfjcommu where AName = ''张三'''

AConnection

adOpenStatic

adLockOptimistic

adCmdText );

memo1.lines.add('********张三的内容如下********');

memo1.lines.add( '共有' + IntToStr( ARecordSet.RecordCount ) + '条匹配的记录' );

{显示各个域的内容}

while not ARecordSet.eof do

begin

strTemp := '';

for intIndex := 0 to ARecordSet.Fields.count - 1 do

strTemp := strTemp + ARecordSet.Fields[intIndex].value+';';

memo1.lines.add( strTemp );

ARecordSet.MoveNext;//移到下条,Next

end;

{关闭数据集和数据库连接}

ARecordSet.close;

AConnection.close;

end;

听别人说总不如自己看实际的例子来体会。在这个例子中,将演示如何利用ADO对象来对一个数据表进行查询、增加记录、修改记录和删除记录操作。具体的用法请参见程序中的注释,如果有点Delphi数据库编程经验,相信不难理解。

在我们的例子使用的数据库为Test.MDB,其中有一个数据表为wfjcommu,有五个字段AName、Portable、Tel、BP、PostAddress,分别表示姓名、手机号、电话号码、呼机号码和通信地址。

以上程序在PWIN98+ Delphi 3.0+PWS(Personal Web Server)4.0下调试通过。

欧朋 CTO 罗志宇:谈Sphinx游戏引擎的前世今生 John Carmack大神亲手操刀,Oculus Rift SDK即将发布 从南极之争谈软件架构十个技巧,及成功团队具备的气质 Imation重拳出击,推出新型混合存储平台 张建华:FusionSphere不是简单的OpenStack发行版 Chrome 29来了:地址栏变得更“兼容” Android版增WebRTC 工程师谈在Facebook工作最糟糕的事情 用Scala重构19楼——SDCC 2013讲师王治专访 10款最新且超实用的开发框架 创业公司为什么应该写博客? 向iOS SDK看齐:Facebook Android SDK迎来重大更新 闯向海外:对话AppFlood产品总监陈霄 高手爱跨界:联想智能桌面,27寸大Pad抢先试玩 惠普第三季度收入272亿美元同比下降8% 高管重新布局 原来我们会出这个价钱购买“廉价”iPhone Web设计过程中该做和不该做的 新细节披露:通过电信合作 美国75%网络遭NSA监控 实时Bug检测工具Bugsnag发布API更新 【观察】一大三小:亚马逊,微软、谷歌、IBM ,IaaS四足鼎立 “化云为雨”一小滴:北京,3G手机+蓝牙打印机实现智能抄表 金山云与蓝汛联合提供云基础设施服务 MongoDB出新招,增加对Hive的支持 编程语言收入榜 Java排第三 直接拿来用!10段超有用的Git命令行代码 Google Cast SDK正式版发布:自动适配,修复Bug 频受红杉资本青睐,商业智能初创公司Birst靠的是什么? 2013百度世界大会要点精华回顾 WebApp开发框架Clouda、轻应用... 对新手有用的JavaScript开发小建议 磨剑三载,且看从容应对14.3万TPS的Twitter是怎样炼成的! 兰亭集势:被高估的“长尾式采购” 谁说WP手机不行?微软称已成拉丁美洲第二大移动平台 图片数据 怎样才能写出即能在VB里又能在ASP里用的COM? 用vc做游戏一般会不会用到MFC 请高手帮忙看看啊,设定SERVICE启动时用户和密码的问题,急啊,47分(我只有这么多可送)!!! 程序员考试 分数是多少? 如何使用属性页propertypage 用JAVA作应用程序服务器,使用SOCKET,如何定义通讯协议? 求助:TClientDataSet的使用(急)!!! 看一看,瞧一瞧啦!!评选近几年中国十大杰出人物!自由提名!!我提两个 先…… IE问题 急!! 请教如何大开Vb3.0以前文件 我要在“1.3.5.2.4.6.3”中看第7位是不是2,该怎么做? 有关HTML中
的问题
嘎毛毛奇怪了. 请教大家一个问题,请看下面。。。 告诉大家一个链接,来着有分,呵呵 请教三层开发 请问用stringGrid控件,怎样获知当前行是第几条纪录? JScrollbar的颜色怎么改??? 请Norwaywoods(挪威的森林)来拿分!!!(方法见内容) php连mssql,php.ini or httpd.conf需要改动什么地方? 关于多文档编程的超难问题,高手请进!!一定给分,我会继续加分!!! 提问:关于软件测试的若干问题,谢绝灌水者入内!高薪聘请业内人士! 请问用javascript怎么显示客户端的IP地址?谢谢 sql问题 oracle中什么权限,才能使一个用户下的表参考另一个用户下的表(就是外键)? 种了中国一号病毒,怎样才能铲草除根? yinx(yinx)请进,拿你应得的分数 请各位高手帮忙,有高分送上。 一个SQL语句的问题。谢啦 为什么我的msdn帮助系统“目录”、“索引”等里的标题都显示不出来了呢? 存储过程只能用一种方式调用吗? 数据库字段问题 如何在ADO的SQL控件中不用ODBC完成长型备注型字段的模糊查询 数据库连接问请各位指点(300分) 圖像控件和報表製作 创天中文版是怎么汉化的?我想把它改回英文的要怎么改? 请问有没有Linux版本的QQ? 找DriverStudio2.5正式版 VB6中不能使用ACCESS2000数据库??? vb中怎么调用dos命令 请问海星... 哪有visual age3.0下载? 有关FTP 我双击注册表编辑器出现"注册表编辑器被管理员禁用"是怎么回事?怎么解决? 如何更改Oracle显示日期的默认方式? *****高手请进****** ======================各位高手,看一下!UP也有分! 请问安徽的高程成绩什么时候可以查? 请问谁知道vb里面如何调用notes对象,怎么调用? 急急急,我好急,有没有办法先把记录集处理一下在绑顶到datagrid -1,4,-7,10…(-1)^n(3n-2)的前n项和等于 设Sn=1*4+2*7+.n(3n+1)则Sn= 燃烧食物时会产生二氧化碳吗 马上要 在线等燃烧食物时会产生二氧化碳吗? 开发橡胶配方用什么显微镜有谁搞这方面的,是观察橡胶中小料的分散程度,大家开发配方都用什么显微镜啊,体式的还是金相的, Sn=1*2+4*2^2+7*2^3+.+(3n-2)*2n^n 用数学归纳法证明1×4+2×7+3×10+……+n(3n+1)=n(n+1) 从事橡胶配方有什么证书吗比如说资格证 室温硫化液体硅橡胶的技术参数与指标各是多少?下面@令箭海棠回答的不错,表示衷心感谢;但是 gb.szrl.net 上的答案更准确一些。 用数学归纳法证明:1.4+2.7+3.10+.+n(3n+1)=n(n+1)² 设计试验方案证明食品充气包装所充气体是什么?小明和小红对市场上销售的一种分袋包装的蛋糕发生了兴趣,因为蛋糕包装在充满气体的小塑料袋内,袋内的气体充得鼓鼓的,看上去好像一个小 为什么东西燃烧后会产生那么大的热量! 用归纳法证明:1*4+2*7+3*10+.+n(3n+1)=n(n+1)^2(n为正整数) 二氧化碳为什么不能对食品充气包装? 2+4+6+8+.+2n=?3+6+9+.+3n=? 用数学归纳法证明:1*2+2*3+3*4+.+n(n+1)=1/3n(n+1)(n+2) 求lim(n→∞)[(1^3+4^3+7^3+……+(3n-2)^3]/{[1+4+7+……+(3n-2)]^2} 橡胶遇热会变成什么? 硫化铁与盐酸的反应硫化铁与浓盐酸和稀盐酸有什么不同的反应吗? 证明食品袋内的气体不是二氧化碳的实验 锡遇热会变形(或散发有毒物质)吗? 亚硫化铁和氧气高温下反应方程式? 如何从一包充气包装的食品袋中取出气体进行检验 995结构胶遇热后有毒吗 硫化氨和氧气怎么反应 橡胶颗粒怎么做阻燃检测 如何验证食品袋内的气体是CO2、N2、稀有气体?要步骤及实验现象. p.s.如果是用燃着的木条的话,其他的不是也会熄灭吗?如二氧化碳等 不会使澄清石灰水变浑浊的也有很多气体吧.怎么判断就是 求证:1*2+2*5+3*8+…+n(3n-1)=n^2(n+1) 4/(1*2*3)+7/(2*3*4)+.+(3n+1)/[n(n+1)(n+2)]急吖还有几个1+1/(1+2)+1/(1+2+3)+.+1/(1+2+3+...+n)=1/(根号下1+根号下2)+1/(根号下2+根号下3)+1/(根号下3+根号下4)+...+1/(根号下n+根号下n+1)=1/3+1/(3+5)+1/(3+5+7)+.+1/(3+5+7+...+2n+1) 平板硫化机都有什么型号的?都是怎么分的? 不同燃料燃烧释放能量的能力是否相同 请简述如何用干燥称重法测食品中的水分.请简述如何用干燥称重法测样品中的水分. 平板硫化机怎么工作的 求证(1/n+1)+(1/n+2)+……(1/3n)>(5/6)[n>=2] 用直接干燥法测定食品中的水分,有时加海沙为的是什么? m-n=2 2m+3n=十四 二元一次方程组 要详解 f(n)=3n^2-3n+1 求证1/f(1)+1/f(2)+1/f(3)+1/f(4)+1/f(5)+…+1/f(n) 直接干燥法测定水分的原理是什么?若用此方法测定半固体或液体样品水分,样品应如何处理? 平板硫化机有什么应用? 1.N+3N-12>或=8 2.2n>-8 3.3n+3 食品水分检测,常压干燥法和甲苯蒸馏法的比较我们公司生产食用香精,涉及的原料及成品时常需要检测水分,一般均食用GB 5009.3的方法检测水分,有时候是常压干燥法,有时候是采用蒸馏法.在检 解方程二元一次方程5分之m-2分之n=2 2m+3n=4 求证:n∈N*,3×5∧2n+1+2∧3n+1可被17整除,在线等啊 生物制品中水分的测定 直接干燥法实验原理及步骤 m+n/2-2m-n/8=2 2m+m-3n/7=-6 二元一次方程 过氧化氢与硫化铁的反应?过氧化氢与硫化铁的反应(H2O2+FeS2)想大家请教一下,硫化铁与过氧化氢的详细反应,最好是反应式,不是电离式.急需,谢谢大家! 反应中是否需要水加入?H2O2+FeS2=? 食品中,自由水和结合水的区别是什么? m-(n+20)÷2=2,2m+3n=12 二元一次方程求了 硫化铁和氧气在高温下反应为什么产生氧化铁,而不产生四氧化三铁 水分测定仪测的水分包括自由水和结合水吗? 二元一次方程 用代入法解下列方程组 m-n/2=2 2m+3n=12 硫化铁为什么比氯化铁更易催化过氧化氢 1+4+7+10+.+(3n-2) 的前n项和 橡胶能不能用硫化机做出各种颜色,可以的话,市面上加工, 硫化铁溶液中加入用硫酸酸化的双氧水溶液的离子方程式 1+1, 1/a +4 , 1/a^2 +7, 1/a^3+10.,1/a^(n-1) + (3n-2) 的前n项和 硫化机是属于橡胶产品制造机械吗?最好详细点儿 食物燃烧时也会生成二氧化碳和水,同时放出热量,这与呼吸作用相类似,你能比较两者的差异吗?要求:两者的场所 反应程度 反应条件 那里有卖pvc排水胶跟给水胶配方与制做 因式分解 1、(3n+2)的平方-25n的平方 2、(x的平方-4x)的平方+8(x的平方-4x)+16 食物燃烧与呼吸作用的差别食物燃烧也会生成CO2和H2O ,同时放出热量 ,这与呼吸作用相类似~你能比较两者的差异吗?
备案号:鲁ICP备13029499号-2 说三道四 www.s3d4.cn