在visual c++中用ado进行数据库编程
作者:蒋东宇
activex数据对象(ado)是ole db上面的高层数据库api。我们在c++程序中也可以调用ado。本文将在vc 6.0环境下做一个小小的例子解释如何使用ado。
1. 生成应用程序框架并初始化ole/com库环境
创建一个标准的mfc appwizard(exe)应用程序,然后在应用程序类的initinstance函数中初始化ole/com库(因为ado库是一个com dll库)。
bool cadotestapp::initinstance()
{ //初始化ole/com库环境
afxoleinit();}
2. 引入ado库文件
使用ado前必须在工程的stdafx.h文件里用直接引入符号#import引入ado库文件,以使编译器能正确编译。代码如下:
#include 〈comdef.h〉
#import "c:\program files\common files\system\ado\msado15.dll"
no_namespace
rename ("eof","adoeof")
头文件comdef.h使我们的应用程序能够使用visual c++中的一些特殊com支持类,这些类使得处理ole自治更为容易一些,ole自治是ado使用的数据类型。后三行使用#import指令在我们的应用程序中输入ado类库定义。
ado类的定义是作为一种资源存储在ado dll(msado15.dll)中,在其内部称为类型库。类型库描述了自治接口,以及c++使用的com vtable接口。当使用#import指令时,在运行时visual c++需要从ado dll中读取这个类型库,并以此创建一组c++头文件。这些头文件具有.tli 和.tlh扩展名,读者可以在项目的目录下找到这两个文件。在c++程序代码中调用的ado类要在这些文件中定义。
程序的第三行指示ado对象不使用名称空间。在有些应用程序中,由于应用程序中的对象与ado中的对象之间可能会出现命名冲突,所以有必要使用名称空间。如果要使用名称空间,则可把第三行程序修改为: rename_namespace("adons")。第四行代码将ado中的eof(文件结束)更名为adoeof,以避免与定义了自己的eof的其他库冲突。
3.利用智能指针进行数据库操作
在caboutdlg头文件中定义两个ado智能指针类实例,并在对话框中加入一个listctrl。
_connectionptr m_pconnection;
_recordsetptr m_precordset;
clistctrl m_list;
ado库包含三个智能指针:_connectionptr、_commandptr和_recordsetptr。
_connectionptr通常被用来创建一个数据连接或执行一条不返回任何结果的sql语句,如一个存储过程。
_commandptr返回一个记录集。它提供了一种简单的方法来执行返回记录集的存储过程和sql语句。在使用_commandptr接口时,可以利用全局_connectionptr接口,也可以在_commandptr接口里直接使用连接串。
_recordsetptr是一个记录集对象。与以上两种对象相比,它对记录集提供了更多的控制功能,如记录锁定、游标控制等。
在oninitdialog()中加入以下代码:
bool caboutdlg::oninitdialog()
{
cdialog::oninitdialog();
_variant_t thevalue;
m_list.resetcontent();
m_pconnection.createinstance(_uuidof(connection));
m_precordset.createinstance(_uuidof(recordset));
try{
m_pconnection->open("dsn=adotest","","",0); //连接叫作adotest的odbc数据源
m_precordset->open("select * from blockdefine",(idispatch*)m_pconnection,
adopendynamic,
adlockoptimistic,
adcmdtext);
//执行sql语句得到一个记录集
while(!m_precordset->adoeof)
//遍历所有记录
{
thevalue = m_precordset->getcollect("blockindex");
//得到字段blockindex的值
if(thevalue.vt!=vt_null)
m_list.addstring((char*)_bstr_t(thevalue)); //将该值加入到列表控件中
m_precordset->movenext();
}
m_precordset->close();
m_pconnection->close();
}
catch(_com_error e) //异常处理
{
afxmessagebox(e->errormessage());
}
m_precordset = null;
m_pconnection = null;
return true; // return true unless you set the focus to a control
}
程序中通过_variant_t和_bstr_t转换com对象和c++类型的数据, _variant_t类封装了ole自治variant数据类型。在c++中使用_variant_t类要比直接使用variant数据类型容易得多。
好,编译后该程序就能运行了,但记住运行前要创建一个叫adotest的odbc数据源。该程序将把表blockdefine中的blockindex字段值显示在列表控件中。