我在《VC知识库在线杂志》第十四期和第十五期上曾发表了两篇文章——“直接通过ODBC读、写Excel表格文件”和“直接通过DAO读、写Access文件”,先后给大家介绍了ODBC和DAO两种数据库访问技术的基本使用方法,这次要给大家介绍的是ADO数据库访问技术的使用方法。ADO(Active Data Object,活动数据对象)实际上是一种基于COM(组件对象模型)的自动化接口(IDispatch)技术,并以OLE DB(对象连接和镶入的数据库)为基础,经过OLE DB精心包装后的数据库访问技术,利用它可以快速的创建数据库应用程序。 ADO提供了一组非常简单,将一般通用的数据访问细节进行封装的对象。由于ODBC数据源也提供了一般的OLE DB Privider,所以ADO不仅可以应用自身的OLE DB Privider,而且还可以应用所有的ODBC驱动程序。关于OLE DB和ADO的其它详细情况,读者可以自行查阅相关书籍或MSDN,这里就不一一说明了。让我们直接步入主题:如何掌握ADO这种数据库访问技术。ADO的操作方法和前面讲过的DAO的操作在很多方面存在相似之处,在这里,笔者为了更有效的说明它的使用方法,用VC6.0做了一个示例程序――AdoRWAccess,这个示例程序可以直接通过ADO来操作Access数据库,示例程序的运行效果如下图所示:

在示例程序中我们仍采用原库结构,数据库名Demo.mdb,库内表名DemoTable,表内字段名为Name(姓名)和Age(年龄)的两个字段,来构造示例程序操作所需的Access数据库,这也和上两篇文章的示例源码中的库结构相兼容。

下面让我们看看ADO数据库访问技术使用的基本步骤及方法:

首先,要用#import语句来引用支持ADO的组件类型库(*.tlb),其中类型库可以作为可执行程序(DLL、EXE等)的一部分被定位在其自身程序中的附属资源里,如:被定位在msado15.dll的附属资源中,只需要直接用#import引用它既可。可以直接在Stdafx.h文件中加入下面语句来实现:

1.#import "c:\program files\common files\system\ado\msado15.dll" \
2.no_namespace \
3.rename ("EOF""adoEOF")

其中路径名可以根据自己系统安装的ADO支持文件的路径来自行设定。当编译器遇到#import语句时,它会为引用组件类型库中的接口生成包装类,#import语句实际上相当于执行了API涵数LoadTypeLib()。#import语句会在工程可执行程序输出目录中产生两个文件,分别为*.tlh(类型库头文件)及*.tli(类型库实现文件),它们分别为每一个接口产生智能指针,并为各种接口方法、枚举类型,CLSID等进行声明,创建一系列包装方法。语句no_namespace说明ADO对象不使用命名空间,rename ("EOF", "adoEOF")说明将ADO中结束标志EOF改为adoEOF,以避免和其它库中命名相冲突。

其次,在程序初始过程中需要初始化组件,一般可以用CoInitialize(NULL);来实现,这种方法在结束时要关闭初始化的COM,可以用下面语句CoUnInitialize();来实现。在MFC中还可以采用另一种方法来实现初始化COM,这种方法只需要一条语句便可以自动为我们实现初始化COM和结束时关闭COM的操作,语句如下所示: AfxOleInit();

接着,就可以直接使用ADO的操作了。我们经常使用的只是前面用#import语句引用类型库时,生成的包装类.tlh中声明的智能指针中的三个,它们分别是_ConnectionPtr、_RecordsetPtr和_CommandPtr。下面分别对它们的使用方法进行介绍:

1、_ConnectionPtr智能指针,通常用于打开、关闭一个库连接或用它的Execute方法来执行一个不返回结果的命令语句(用法和_CommandPtr中的Execute方法类似)。
――打开一个库连接。先创建一个实例指针,再用Open打开一个库连接,它将返回一个IUnknown的自动化接口指针。代码如下所示:

01._ConnectionPtr  m_pConnection;
02.// 初始化COM,创建ADO连接等操作
03.AfxOleInit();
04.m_pConnection.CreateInstance(__uuidof(Connection));
05. 
06.// 在ADO操作中建议语句中要常用try...catch()来捕获错误信息,
07.// 因为它有时会经常出现一些意想不到的错误。jingzhou xu
08.try                
09.{  
10.// 打开本地Access库Demo.mdb
11.m_pConnection->Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=Demo.mdb","","",adModeUnknown);
12.}
13.catch(_com_error e)
14.{
15.AfxMessageBox("数据库连接失败,确认数据库Demo.mdb是否在当前路径下!");
16.return FALSE;
17.}

――关闭一个库连接。如果连接状态有效,则用Close方法关闭它并赋于它空值。代码如下所示:

1.if(m_pConnection->State)
2.m_pConnection->Close();
3.m_pConnection= NULL;

2、_RecordsetPtr智能指针,可以用来打开库内数据表,并可以对表内的记录、字段等进行各种操作。――打开数据表。打开库内表名为DemoTable的数据表,代码如下:

01._RecordsetPtr   m_pRecordset;
02.m_pRecordset.CreateInstance(__uuidof(Recordset));
03. 
04.// 在ADO操作中建议语句中要常用try...catch()来捕获错误信息,
05.// 因为它有时会经常出现一些意想不到的错误。jingzhou xu
06.try
07.{
08.m_pRecordset->Open("SELECT * FROM DemoTable",                // 查询DemoTable表中所有字段
09.theApp.m_pConnection.GetInterfacePtr(),  // 获取库接库的IDispatch指针
10.adOpenDynamic,
11.adLockOptimistic,
12.adCmdText);
13.}
14.catch(_com_error *e)
15.{
16.AfxMessageBox(e->ErrorMessage());
17.}

――读取表内数据。将表内数据全部读出并显示在列表框内,m_AccessList为列表框的成员变量名。如果没有遇到表结束标志adoEOF,则用GetCollect(字段名)或m_pRecordset->Fields->GetItem(字段名)->Value方法,来获取当前记录指针所指的字段值,然后再用MoveNext()方法移动到下一条记录位置。代码如下所示:

01._variant_t var;
02.CString strName,strAge;
03.try
04.{
05.if(!m_pRecordset->BOF)
06.m_pRecordset->MoveFirst();
07.else
08.{
09.AfxMessageBox("表内数据为空");
10.return;
11.}
12. 
13.// 读入库中各字段并加入列表框中
14.while(!m_pRecordset->adoEOF)
15.{
16.var = m_pRecordset->GetCollect("Name");
17.if(var.vt != VT_NULL)
18.strName = (LPCSTR)_bstr_t(var);
19.var = m_pRecordset->GetCollect("Age");
20.if(var.vt != VT_NULL)
21.strAge = (LPCSTR)_bstr_t(var);
22. 
23.m_AccessList.AddString( strName + " --> "+strAge );
24. 
25.m_pRecordset->MoveNext();
26.}
27. 
28.// 默认列表指向第一项,同时移动记录指针并显示
29.m_AccessList.SetCurSel(0);
30.}
31.catch(_com_error *e)
32.{
33.AfxMessageBox(e->ErrorMessage());
34.}

――插入记录。可以先用AddNew()方法新增一个空记录,再用PutCollect(字段名,值)输入每个字段的值,最后再Update()更新到库中数据既可。其中变量m_Name和m_Age分别为姓名及年龄编辑框的成员变量名。代码所下所示:

01.try
02.{
03.// 写入各字段值
04.m_pRecordset->AddNew();
05.m_pRecordset->PutCollect("Name", _variant_t(m_Name));
06.m_pRecordset->PutCollect("Age"atol(m_Age));
07.m_pRecordset->Update();
08. 
09.AfxMessageBox("插入成功!");
10.}
11.catch(_com_error *e)
12.{
13.AfxMessageBox(e->ErrorMessage());
14.}

――移动记录指针。移动记录指针可以通过MoveFirst()方法移动到第一条记录、MoveLast()方法移动到最后一条记录、MovePrevious()方法移动到当前记录的前一条记录、MoveNext()方法移动到当前记录的下一条记录。但我们有时经常需要随意移动记录指针到任意记录位置时,可以使用Move(记录号)方法来实现,注意: Move()方法是相对于当前记录来移动指针位置的,正值向后移动、负值向前移动,如:Move(3),当前记录是3时,它将从记录3开始往后再移动3条记录位置。代码如下所示:

01.try
02.{
03.int curSel = m_AccessList.GetCurSel(); 
04.// 先将指针移向第一条记录,然后就可以相对第一条记录来随意移动记录指针
05.m_pRecordset->MoveFirst();
06.m_pRecordset->Move(long(curSel));
07. 
08.}
09.catch(_com_error *e)
10.{
11.AfxMessageBox(e->ErrorMessage());
12.}

――修改记录中字段值。可以将记录指针移动到要修改记录的位置处,直接用PutCollect(字段名,值)将新值写入并Update()更新数据库既可。可以用上面方法移动记录指针,修改字段值代码如下所示:

01.try
02.{
03.// 假设对第二条记录进行修改
04.m_pRecordset->MoveFirst();
05.m_pRecordset->Move(1);        // 从0开始
06.m_pRecordset->PutCollect("Name", _variant_t(m_Name));
07.m_pRecordset->PutCollect("Age"atol(m_Age));
08.m_pRecordset->Update();
09.}
10.catch(_com_error *e)
11.{
12.AfxMessageBox(e->ErrorMessage());
13.}

――删除记录。删除记录和上面修改记录的操作类似,先将记录指针移动到要修改记录的位置,直接用Delete()方法删除它并用Update()来更新数据库既可。代码如下所示:

01.try
02.{
03.// 假设删除第二条记录
04.m_pRecordset->MoveFirst();
05.m_pRecordset->Move(1);        // 从0开始
06.m_pRecordset->Delete(adAffectCurrent);  // 参数adAffectCurrent为删除当前记录
07.m_pRecordset->Update();
08.}
09.catch(_com_error *e)
10.{
11.AfxMessageBox(e->ErrorMessage());
12.}

――关闭记录集。直接用Close方法关闭记录集并赋于其空值。代码如下所示:

1.m_pRecordset->Close();
2.m_pRecordset = NULL;

3、CommandPtr智能指针,可以使用_ConnectionPtr或_RecordsetPtr来执行任务,定义输出参数,执行存储过程或SQL语句。 
――执行SQL语句。先创建一个_CommandPtr实例指针,再将库连接和SQL语句做为参数,执行Execute()方法既可。代码如下所示:

1._CommandPtr     m_pCommand;
2.m_pCommand.CreateInstance(__uuidof(Command));
3.m_pCommand->ActiveConnection = m_pConnection;  // 将库连接赋于它
4.m_pCommand->CommandText = "SELECT * FROM DemoTable";  // SQL语句
5.m_pRecordset = m_pCommand->Execute(NULL, NULL,adCmdText); // 执行SQL语句,返回记录集

――执行存储过程。执行存储过程的操作和上面执行SQL语句类似,不同点仅是CommandText参数中不再是SQL语句,而是存储过程的名字,如Demo。另一个不同点就是在Execute()中参数由adCmdText(执行SQL语句),改为adCmdStoredProc来执行存储过程。如果存储过程中存在输入、输出参数的话,需要使用到另一个智能指针_ParameterPtr来逐次设置要输入、输出的参数信息,并将其赋于_CommandPtr中Parameters参数来传递信息,有兴趣的读者可以自行查找相关书籍或MSDN。执行存储过程的代码如下所示:

1._CommandPtr     m_pCommand;
2.m_pCommand.CreateInstance(__uuidof(Command));
3.m_pCommand->ActiveConnection = m_pConnection;  // 将库连接赋于它
4.m_pCommand->CommandText = "Demo"
5.m_pCommand->Execute(NULL,NULL, adCmdStoredProc);

最后,如果想知道详细实现细节的话,可以在下载示例源码后,仔细查看源码既可(内有详细注释)。

直接通过ADO操作Access数据库的更多相关文章

  1. MFC通过ADO操作Access数据库

    我在<VC知识库在线杂志>第十四期和第十五期上曾发表了两篇文章——“直接通过ODBC读.写Excel表格文件”和“直接通过DAO读.写Access文件”,先后给大家介绍了ODBC和DAO两 ...

  2. ADO访问Access数据库错误解决心得随笔

    最近在用ADO访问Access数据库的时候出现了一个奇怪的错误,觉得有必要记录下来,和大家分享一下. 环境 win7 x86系统: VS2012编译器: Office2010: Access2000~ ...

  3. Python操作Access数据库

    我们在这篇文章中公分了五个步骤详细分析了Python操作Access数据库的相关方法,希望可以给又需要的朋友们带来一些帮助. AD: Python编 程语言的出现,带给开发人员非常大的好处.我们可以利 ...

  4. 关于操作Access数据库jdk选择问题

    关于操作Access数据库,使用jdk64位无法通过ODBC无法获取数据,只能通过jdk32位进行开发.

  5. 基于指纹考勤机的真实的PHP操作Access数据库成功案例(最终实现) 2011-11-2v

    听了我的建议,我们单位的食堂准备使用一台指纹考勤机统计吃饭人次,这样院里好给食堂的承包人以相应饭补.以前买过一台彩屏指纹机,数据库是access的,今儿又买了一台准备放到食堂里,而且考虑到停电,还特地 ...

  6. x64 win64编译环境下ADO链接Access数据库的问题解决

    原文链接地址:https://blog.csdn.net/HW140701/article/details/71077579 Win32编译环境下,用ADO数据库连接Access数据库一般都不会报错, ...

  7. JavaScript操作数据库JS操作Access数据库

    avaScript操作数据库JS操作Access数据库,跟其他语言操作差不多,总结了一下习惯代码,仅供参考学习.现在在F盘有文件abc.mdf,表名为Student,一共2个字段,Id数字类型主键,s ...

  8. VC++中使用ADO方式操作ACCESS数据库

    ADO(ActiveX Data Object)是Microsoft数据库应用程序开发的新接口,是建立在OLE DB之上的高层数据库访问技术,即使你对OLE DB,COM不了解也能轻松对付ADO,因为 ...

  9. 【c#】ADO操作Access的mdb数据库只能读不能修改的解决方法

    在使用ACCESS数据库时连接字符串如 string strcon=@"Provider=Microsoft.Jet.OLEDB.4.0;Data Source=F:\Access操作\简易 ...

随机推荐

  1. asp.net Core 部署到CentOs7上,使用Nginx做代理

    一.CentOs7部署Nginx 1.准备工作 Nginx的安装依赖于以下三个包,意思就是在安装Nginx之前首先必须安装一下的三个包,注意安装顺序如下: 1 SSL功能需要openssl库,直接通过 ...

  2. No module named yum

    升级python之后,执行yum的时候可能出现错误,类似: There was a problem importing one of the Python modulesrequired to run ...

  3. linux下{}的用法

    在touch {a,b}.txt时,同时创建了a.txt,b.txt两个文件 而touch {1..10}.txt,同时创建了10个txt文件,从1.txt到10.txt 在linux通配符中,{n, ...

  4. LeetCode OJ:Search Insert Position(查找插入位置)

    Given a sorted array and a target value, return the index if the target is found. If not, return the ...

  5. java中商业数据计算时用到的类BigDecimal和DecimalFormat

    1.引言 借用<Effactive Java>这本书中的话,float和double类型的主要设计目标是为了科学计算和工程计算.他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确 ...

  6. Week04《Java程序设计》第四周学习总结

    Week04<Java程序设计>第四周学习总结 1. 本周学习总结 1.1 写出你认为本周学习中比较重要的知识点关键词 答:static关键字,final关键字,静态初始块,抽象类,继承, ...

  7. 深入理解UE4宏定义—— GENERATED_BODY

    本文章由cartzhang编写,转载请注明出处. 所有权利保留. 文章链接:http://blog.csdn.net/cartzhang/article/details/72834164 作者:car ...

  8. Android 进阶7:进程通信之 AIDL 的使用

    读完本文你将了解: AIDL 是什么 AIDL 支持的数据类型 AIDL 如何编写 AIDL 实例 创建 AIDL 编写服务端代码 编写客户端代码 运行结果 总结 代码地址 Thanks 记得 201 ...

  9. Unity3d command line arguments

    Options Many of these relate to Unity3d command line arguments Batch Mode - should be left enabled u ...

  10. Ubuntu 16.04 为 PHP7 添加 memcached 以及 redis 扩展

    切换到 PHP 7 之后,网站的速度大幅提升,不过通常的扩展可能某一个就还没有支持 PHP7 Memcached 比如说我现在使用了最新的 Ubuntu 16.04,虽然内置了 PHP 7 源,但 m ...