一、数据库操作准备
1、导入ADO动态链接库
在工程的stdafx.h中加入如下语句:
#import "c:\program files\common
files\system\ado\msado15.dll" no_namespace\
rename("EOF","adoEOF")
这一语句有何作用呢?其最终作用同我们熟悉的#include类似,编译的时候系统会为我们生成msado15.tlh和msado15.tli两个C++头文件来定义ADO库,即加载ADO动态库(msado15.dll)。
其中,no_namespace表明不使用命名空间,rename("EOF","adoEOF")表明把ADO中用到的EOF改为adoEOF,防止发生命名冲突。
注意:该代码需要在一行中完成,如果写成两行或者多行,行末要加上“\”符号,表示把这几行看成一行,如本例。
2、初始化OLE/COM库环境
在基于MFC的应用里,初始化OLE/COM库环境的一个比较好的位置是在应用类的InitInstance成员函数中,而且直接使用AfxOleInit,在退出应用时,该函数也负责COM资源的释放,将此函数添加在InitInstance中的如下位置:
BOOL
CExpApp::InitInstance()
{
AfxEnableControlContainer();
//初始化OLE DLLs
if(!AfxOleInit())
{
AfxMessageBox("初始化OLE DLL失败!");
Return FALSE;
}
......
}
说明:也可以在InitInstance中使用::CoInitialize初始化OLE/COM库环境,但须在ExitInitInstance中使用::CoUninitialize释放占用的COM资源,显然使用AfxOleInit更为方便。
3、连接数据库
在Doc\View程序中,通常在应用类(App类)中进行数据库的连接。
1)声明一个Connection指针
_ConnectionPtr
m_pConnection;
注:ADO最重要的三个对象有三个:连接对象(Connection)、命令对象(Command)和记录集对象(RecordSet)。在使用这三个对象的时候,需要定义与之相对应的智能指针,分别为_ConnectionPtr、_CommandPtr、_RecordsetPtr。
由上述ConnectionPtr指针的使用步骤可知,和C++中的类指针使用方法一样,智能指针也要先定义指针变量、创建其实例(实例化),然后就可以调用它的方法和属性。不同的是,该智能指针最后是自动进行内存释放的。
所有的智能指针都是基于_com_ptr_t模板类的,该类封装了IUnknow接口的3个方法:QueryInterface、Addref和Release。它具有自动计数的机制,即在构造对象时,自动为该对象计数加1。析构对象时,自动调用Release方法。(即该类型的指针在使用后不需要手动释放内存)(但需要调用Close方法,关闭连接或者记录集)所以智能指针会使代码更加简洁并且不易出错。
2)创建Connection对象
m_pConnection.CreateInstance(__uuidof(Connection));
m_pConnection.CreateInstance("ADODB.Connection");
上述两种方法均可。
注意:上面调用_ConnectionPtr接口指针的方法CreateInstance时,用的是“.”而非
“->”。
3)设置连接字符串,以便指定需要的连接
3.1) 使用JET数据库引擎实现对Acess2000类型的数据库info.mdb的连接
CString
strSQL="Provider=Microsoft.Jet.OLEDB.4.0;Data
Source=info.mdb;User ID=admin;Passward=;";
或者
CString
strSQL=_T("Provider=Microsoft.Jet.OLEDB.4.0;Data
Source=info.mdb;User ID=admin;Passward=;");
3.2)
使用OLE DB提供者实现对SQL Server的标准安全连接串
strConnect=_T("Provider=sqloledb;Data
Source=MyServerName;"
"Initial Catalog=MyDateBaseName;"
"User ID=MyUserName;Password=MyPassword;");
例程:strConnection="Provider=SQLOLEDB;DataSource=local;InitialCatalog=DVDRentDB_Data.MDF;"
"User ID=sa;Password=820415";
m_pConnection->Open((_bstr_t)strSQL,"","",adModeUnknown);
或者是在此处不设置User ID和Password,而直接在Open的第2、3个参数中设置。
strConnection="Provider=SQLOLEDB;DataSource=local;InitialCatalog=DVDRentDB_Data.MDF";
m_pConnection->Open((_bstr_t)strSQL,"sa","820415",adModeUnknown);
注意:上面设置连接字符串的时候,如果过长需要分行时,则每一行都要加上双引号,在最后加上分号即可。
如果是本地服务器,则DataSource=local或本地服务器名均可
若数据库没有设置密码,在连接字符串中可以将其省略,但User ID不能省
若数据库和程序文件不在同一文件夹下,直接写数据库名即可,在InitialCatalog中不需加上该数据库的存储器地址
3.3)
使用OLE DB提供者实现对远程SQL Server的标准安全连接串
strConnect=_T("Provider=sqloledb;Network
Library=DBMSSOCN;"
"Data Source=130.120.110.001,1433;"
"Initial Catalog=MyDateBaseName;"
"User ID=MyUserName;Password=MyPassword;");
4)、使用m_pConnection的Open方法实现对数据库的连接
在ADO的操作中建议使用try...catch(
)来捕获错误信息,因为它有时会经常出现一些意想不到的错误
try
{
m_pConnection->Open( (_bstr_t) strSQL," ","
",adModeUnknown);
}
catch(_com_error
e)
//捕捉异常
{
CString strError;
strError.Format( "连接数据库发生异常! \r \n错误信息:%s",e.ErrorMessage( )
);
AfxMessageBox(errormessage);
//显示错误信息
}
4、关闭连接
一般重载App类的ExitInstace( )函数实现
调用m_pConnection的Close方法关闭连接即可
m_pConnection->Close(
);
m_pConnection=NULL;
注意:由于初始化COM库调用的是AfxOleInit,这种方法初始化COM库的优点就在于资源
的释放也是自动进行的,所以不必担心资源泄漏的问题。
二、数据库操作
ADO库中包含的三个基本接口为_ConnectionPtr接口、_CommandPtr接口、_RecordsetPtr接口。
1、_ConnectionPtr接口
该接口返回一个记录集或一个空指针。
通常用它来创建一个数据库连接,或执行一条不返回任何结果的SQL语句,如一个存储过程。
不推荐使用_ConnectionPtr接口返回一个记录集,对于要返回记录集的操作通常用_RecordsetPtr来实现,而且使用_ConnectionPtr时要想得到记录数目必须遍历所有记录,但使用_RecordsetPtr时则不需要。
2、_CommandPtr接口
该接口返回一个记录集。
它提供了一种简单的方法来执行返回记录集的存储过程和SQL语句。
在使用_CommandPtr接口时,可以利用全局_ConnectionPtr接口,也可以在_CommandPtr
接口里直接使用连接串。如果只执行一次或者几次数据库访问操作,后者是比较好的选择。但是,如果频繁访问数据库,并要返回很多记录集,那么应该使用全局_ConnectionPtr接口创建一个数据库连接,然后使用_CommandPtr接口执行存储过程和SQL语句。
3、_RecordsetPtr接口
该接口是一个记录集对象。
与前两种对象相比,它对记录集提供了更多的控制功能,如记录锁定、游标控制等。同_CommandPtr接口一样,它不一定要使用一个已经创建的数据库连接,可以用一个连接串代替连接指针赋给_RecordsetPtr的connection成员变量,让它自己创建数据库连接。如果使用多个记录集,最好的方法是同Command对象一样使用已经创建了数据连接的全局_ConnectionPtr接口,然后使用_RecordsetPtr执行存储过程和SQL语句。
注意:可以使用Recordset对象来执行查询命令,但如果查询或者存储过程是需要参数的,这时就只能使用Command对象。
使用Recordset对象操作数据库:
假定已经成功使用Connection对象创建了数据源的连接,连接指针为m_pConnection。
1)创建记录集
声明记录集指针
_RecordsetPtr
m_pRecordset;
创建记录集
m_pRecordset.CreateInstance(__uuidof(Recordset));
2)打开记录集
记录集指针创建完毕后,调用该指针的Open方法打开记录集。
该函数声明如下:
HRESULT Recordset15::Open ( const _variant_t &
Source,
const _variant_t & ActiveConnection,
enum CursorTypeEnum CursorType,
enum LockTypeEnum LockType,
long Options ) ;
各个参数的含义如下:
参数Source:为_variant_t类型的引用,可以为有效的Command对象、SQL语句、表名、存储过程调用等。
参数ActiveConnection:为_variant_t类型的引用,为已经建立好的连接。
参数CursorType:用于设置在打开Recordset时提供者应使用的游标类型,它可取CursorTypeEnum
中的任一值,默认值为adOpenForwardOnly。
参数
LockType:用于设置在打开Recordset时提供者应使用的锁定类型,它可取枚举LockTypeEnum中的任一值,默认值为adLockReadOnly。
参数 Options:用于设置获取Source(即Open第一个参数)的方式,其类型long。
http://img.blog.163.com/photo/NKGtN8m2TDFaVOraJ_XhpQ==/1137440380888802474.jpg
http://img.blog.163.com/photo/aYUXNbi2t8Ns0HGnbQIdMg==/1137440380888802475.jpg
http://img.blog.163.com/photo/KmQpxoCEscnhojU0jhGkwQ==/1137440380888802476.jpg
例程1: CString strSQL = "select * from
mytablename";
m_pRecordset->Open
( _variant_t (strSQL),
m_pConnection.GetInterfacePtr(
),
adOpenDynamic,
adLockOptimistic,