正文

数据库编程总结2010-04-21 20:10:00

【评论】 【打印】 【字体: 】 本文链接:http://blog.pfan.cn/lym51/50977.html

分享到:

数据库编程总结 http://blog.csdn.net/byxdaz/archive/2010/04/11/5473557.aspx 当前各种主流数据库有很多,包括Oracle, MS SQL Server, Sybase, Informix, MySQL, DB2, Interbase / Firebird, PostgreSQL, SQLite, SAP/DB, TimesTen, MS ACCESS等等。数据库编程是对数据库的创建、读写等一列的操作。数据库编程分为数据库客户端编程与数据库服务器端编程。数据库客户端编程主要使用ODBC API、ADO、ADO.NET、OCI、OTL等方法;数据库服务端编程主要使用OLE DB等方法。数据库编程需要掌握一些访问数据库技术方法,还需要注意怎么设计高效的数据库、数据库管理与运行的优化、数据库语句的优化。 一、访问数据库技术方法 数据库编程分为数据库客户端编程与数据库服务器端编程。数据库客户端编程主要使用ODBC API、ADO、ADO.NET、OCI、OTL等方法;数据库服务端编程主要使用OLE DB等方法。 1、几种是数据库访问方法比较 ODBC API是一种适合数据库底层开发的编程方法,ODBC API 提供大量对数据源的操作,ODBC API能够灵活地操作游标,支持各种帮定选项,在所有ODBC相关编程中,API编程具有最高的执行速度。 DAO提供了很好的数据库编程的对象模型.但是,对数据库的所有调用以及输出的数据都必须通过Access/Jet 数据库引擎,这对于使用数据库应用程序,是严重的瓶颈。 OLE DB提供了COM接口,与传统的数据库接口相比,有更好的健壮性和灵活性,具有很强的错误处理能力,能够同非关系数据源进行通信。 ADO最主要的优点在于易于使用、速度快、内存支出少和磁盘遗迹小。 ADO.NET 是利用数据集的概念将数据库数据读入内存中,然后在内存中对数据进行操作,最后将数据集数据回写到源数据库中。 OTL 是 Oracle, Odbc and DB2-CLI Template Library 的缩写,是一个C++编译中操控关系数据库的模板库, OTL中直接操作Oracle主要是通过Oracle提供的OCI接口进行,进行操作DB2数据库则是通过CLI接口来进行,至于MS的数据库和其它一些数据库,则OTL只提供了ODBC来操作的方式。当然Oracle和DB2也可以由OTL间接使用ODBC的方式来进行操纵。具有以下优点:跨平台;运行效率高,与C语言直接调用API相当;开发效率高,起码比ADO.net使用起来更简单,更简洁;部署容易,不需要ADO组件,不需要.net framework 等。 2、 VC数据库编程几种方法 VC 数据库编程几种方法,包括ODBC连接、 MFC ODBC连接、DAO连接、OLE DB、OLE DB Templates连接、ADO、Oracle专用方法(OCI(Oracle Call Interface)访问、Oracle Object OLE C++ Class Library )。 通用方法 1. ODBC连接 ODBC(Open DataBase Connectivity)是MSOA的一部分,是一个标准数据库接口。它提供对关系数据库访问的统一接口,实现对异构数据源的一致访问。 ODBC 数据访问由以下部分组成: 句柄(Handles):ODBC使用句柄来标识ODBC环境、连接、语句和描述器. 缓存区(Buffers): 数据类型(Data types) 一致性级别(Conformance levels) 用ODBC设计客户端的一般步骤: 分配ODBC环境 分配连接句柄 连接数据源 构造和执行SQL语句 获得查询结果 断开数据源的连接 释放ODBC环境 ODBC API是一种适合数据库底层开发的编程方法,ODBC API提供大量对数据源的操作,ODBC API能够灵活地操作游标,支持各种帮定选项,在所有ODBC相关编程中,API编程具有最高的执行速度.因此,ODBC API编程属于底层编程。 2. MFC ODBC连接 MFC ODBC是MFC对ODBC进行的封装,以简化对ODBC API的 调用,从而实现面向对象的数据库编程接口. MFC ODBC的封装主要开发了CDatabase类和CRecordSet类 (1) CDatabase类 CDatabase类用于应用程序建立同数据源的连接。CDatabase类中包含一个m_hdbc变量,它代表了数据源的连接句柄。如果要建立 CDatabase类的实例,应先调用该类的构造函数,再调用Open函数,通过调用,初始化环境变量,并执行与数据源的连接。在通过Close函数关闭数据源。 CDatabase类提供了对数据库进行操作的函数及事务操作。 (2) CRecordSet类 CRecordSet类定义了从数据库接收或者发送数据到数据库的成员变量,以实现对数据集的数据操作。 CRecordSet类的成员变量m_hstmt代表了定义该记录集的SQL语句句柄,m_nFields为记录集中字段的个数,m_nParams为记录集所使用的参数个数。 CRecordSet的记录集通过CDatabase实例的指针实现同数据源的连接,即CRecordSet的成员变量m_pDatabase. MFC ODBC编程更适合于界面型数据库应用程序的开发,但由于CDatabase类和CRecordSet类提供的数据库操作函数有限,支持的游标类型也有限,限制了高效的数据库开发。在编程层次上属于高级编程。 应用实例: 1.打开数据库 CDatabase database; database.OpenEx( _T( "DSN=zhuxue" ),CDatabase::noOdbcDialog);//zhuxue为数据源名称 2.关联记录集 CRecordset recset(&database); 3.查询记录 CString sSql1=""; sSql1 = "SELECT * FROM tablename" ; recset.Open(CRecordset::forwardOnly, sSql1, CRecordset::readOnly); int ti=0; CDBVariant var;//var可以转换为其他类型的值 while (!recset.IsEOF()) { //读取Excel内部数值 recset.GetFieldValue("id",var); jiangxiang[ti].id=var.m_iVal; recset.GetFieldValue("name", jiangxiang[ti].name); ti++; recset.MoveNext(); } recset.Close();//关闭记录集 4.执行sql语句 CString sSql=""; sSql+="delete * from 院系审核";//清空表 database.ExecuteSQL(sSql); sSql也可以为Insert ,Update等语句 5.读取字段名 sSql = "SELECT * FROM Sheet1" ; //读取的文件有Sheet1表的定义,或为本程序生成的表. // 执行查询语句 recset.Open(CRecordset::forwardOnly, sSql, CRecordset::readOnly); int excelColCount=recset.GetODBCFieldCount();//列数 CString excelfield[30]; //得到记录集的字段集合中的字段的总个数 for( i=0;i= wsprintf( szInitStr, _T("Provider=SQLOLEDB;Data Source=%s;Initial Catalog=master;User Id=%s;Password=%s;Connect Timeout=%u"), pszDataSource, pszUserID, pszPassword, uTimeout ) ); //Initial Catalog=master指明连接成功后,"USE master"。 EAutoReleasePtr pIDataInitialize; HRESULT hResult = ::CoCreateInstance( CLSID_MSDAINITIALIZE, NULL, CLSCTX_INPROC_SERVER, IID_IDataInitialize, ( void** )&pIDataInitialize ); if( FAILED( hResult ) ) { return hResult; } EAutoReleasePtr pIDBInitialize; hResult = pIDataInitialize->GetDataSource( NULL, CLSCTX_INPROC_SERVER, ( LPCOLESTR )szInitStr, IID_IDBInitialize, ( IUnknown** )&pIDBInitialize ); if( FAILED( hResult ) ) { return hResult; } hResult = pIDBInitialize->Initialize( ); if( FAILED( hResult ) ) { return hResult; } * ppIDBInitialize = pIDBInitialize.Detach( ); return S_OK; } HRESULT CreateSession( IDBInitialize* pIDBInitialize, IOpenRowset** ppIOpenRowset ) { ASSERT( pIDBInitialize != NULL && ppIOpenRowset != NULL ); EAutoReleasePtr pSession; HRESULT hResult = pIDBInitialize->QueryInterface( IID_IDBCreateSession, ( void** )&pSession ); if( FAILED( hResult ) ) { return hResult; } EAutoReleasePtr pIOpenRowset; hResult = pSession->CreateSession( NULL, IID_IOpenRowset, ( IUnknown** )&pIOpenRowset ); if( FAILED( hResult ) ) { return hResult; } * ppIOpenRowset = pIOpenRowset.Detach( ); return S_OK; } HRESULT CreateCommand( IOpenRowset* pIOpenRowset, ICommand** ppICommand, ICommandText** ppICommandText ) { ASSERT( pIOpenRowset != NULL && ppICommand != NULL && ppICommandText != NULL ); HRESULT hResult; EAutoReleasePtr pICommand; { EAutoReleasePtr pICreateCommand; hResult = pIOpenRowset->QueryInterface( IID_IDBCreateCommand, ( void** )&pICreateCommand ); if( FAILED( hResult ) ) { return hResult; } hResult = pICreateCommand->CreateCommand( NULL, IID_ICommand, (IUnknown**)&pICommand ); if( FAILED( hResult ) ) { return hResult; } } EAutoReleasePtr pICommandText; hResult = pICommand->QueryInterface( &pICommandText ); if( FAILED( hResult ) ) { return hResult; } * ppICommand = pICommand.Detach( ); * ppICommandText = pICommandText.Detach( ); return S_OK; } HRESULT ExecuteSQL( ICommand* pICommand, ICommandText* pICommandText, LPCTSTR pszCommand, LONG* plRowsAffected ) { ASSERT( pICommand != NULL && pICommandText != NULL && pszCommand != NULL && pszCommand[0] != 0 ); HRESULT hResult = pICommandText->SetCommandText( DBGUID_DBSQL, ( LPCOLESTR )pszCommand ); if( FAILED( hResult ) ) { return hResult; } LONG lAffected; hResult = pICommand->Execute( NULL, IID_NULL, NULL, plRowsAffected == NULL ? &lAffected : plRowsAffected, ( IUnknown** )NULL ); return hResult; } 以上就是写数据库的全部代码了,是不是很简单呢?下面再来读的。 // 先用与上面代码中一样的步骤获取pICommand,pICommandText。此处省略 HRESULT hResult = pICommandText->SetCommandText( DBGUID_DBSQL, ( LPCOLESTR )_T("SELECT Volume FROM 2005_1 WHERE ID = @@IDENTITY") ); //取我们刚刚添加的那一条记录 if( FAILED( hResult ) ) { return; } LONG lAffected; EAutoReleasePtr pIRowset; hResult = pICommand->Execute( NULL, IID_IRowset, NULL, &lAffected, ( IUnknown** )&pIRowset ); if( FAILED( hResult ) ) { return; } EAutoReleasePtr pIAccessor; hResult = pIRowset->QueryInterface( IID_IAccessor, ( void** )&pIAccessor ); if( FAILED( hResult ) ) { return; } // 一个根据表中各字段的数值类型而定义的结构,用于存储返回的各字段的值 struct CLoadLastFromDB { DBSTATUS dwdsVolume; DWORD dwLenVolume; float fVolume; }; // 此处我们只查询了一个字段。如果要查询多个字段,CLoadLastFromDB中要添加相应的字段定义,下面的dbBinding也要相应扩充。dbBinding[].iOrdinal要分别指向各个字段,dbBinding[].wType要根据字段类型赋合适的值。 DBBINDING dbBinding[1]; dbBinding[0].iOrdinal = 1; // Volume 字段的位置,从 1 开始 dbBinding[0].obValue = offsetof( CLoadLastFromDB, fVolume ); dbBinding[0].obLength = offsetof( CLoadLastFromDB, dwLenVolume ); dbBinding[0].obStatus = offsetof( CLoadLastFromDB, dwdsVolume ); dbBinding[0].pTypeInfo = NULL; dbBinding[0].pObject = NULL; dbBinding[0].pBindExt = NULL; dbBinding[0].dwPart = DBPART_VALUE | DBPART_STATUS | DBPART_LENGTH; dbBinding[0].dwMemOwner = DBMEMOWNER_CLIENTOWNED; dbBinding[0].eParamIO = DBPARAMIO_NOTPARAM; dbBinding[0].cbMaxLen = 0; dbBinding[0].dwFlags = 0; dbBinding[0].wType = DBTYPE_R4; // float就是DBTYPE_R4,int就是DBTYPE_I4。参见MSDN dbBinding[0].bPrecision = 0; dbBinding[0].bScale = 0; HACCESSOR hAccessor = DB_NULL_HACCESSOR; DBBINDSTATUS dbs[1]; hResult = pIAccessor->CreateAccessor( DBACCESSOR_ROWDATA, 1, dbBinding, sizeof( CLoadLastDataFromDB ), &hAccessor, dbs ); if( FAILED( hResult ) ) { return; } ASSERT( dbs[0] == DBBINDSTATUS_OK ); ULONG uRowsObtained = 0; HROW hRows[1]; // 这里我们只查询了最新的那一条记录 HROW* phRows = hRows; CLoadLastFromDB rmd; hResult = pIRowset->GetNextRows( NULL, 0, 1, &uRowsObtained, &phRows ); if( SUCCEEDED( hResult ) && uRowsObtained != 0U ) { hResult = pIRowset->GetData( phRows[0], hAccessor, &rmd ); if( FAILED( hResult ) ) { ASSERT( FALSE ); } ASSERT( rmd.dwdsVolume == DBSTATUS_S_OK ); // rmd.fVolume 就是我们要取的值 } pIRowset->ReleaseRows( uRowsObtained, phRows, NULL, NULL, NULL ); pIAccessor->ReleaseAccessor( hAccessor, NULL ); pIAccessor.Release( ); pIRowset.Release( ); 读操作也完成了,是不是仍然很简单呢?下面我们再来看看最麻烦的二进制数据(text、ntext、image等)的读写。要实现BLOB数据的读写,我们需要一个辅助的类,定义如下: class CSequentialStream : public ISequentialStream // BLOB 数据访问类 { public: CSequentialStream( ); virtual ~CSequentialStream( ); virtual BOOL Seek( ULONG uPosition ); virtual BOOL Clear( ); virtual ULONG GetLength( ) { return m_uBufferUsed; }; virtual operator void* const( ) { return m_pBuffer; }; STDMETHODIMP_( ULONG ) AddRef( ) { return ++ m_uRefCount; }; STDMETHODIMP_( ULONG ) Release( ) { ASSERT( m_uRefCount != 0U ); -- m_uRefCount; if( m_uRefCount == 0U ) { delete this; } return m_uRefCount; }; STDMETHODIMP QueryInterface( REFIID riid, LPVOID* ppv ); STDMETHODIMP Read( void __RPC_FAR* pv, ULONG cb, ULONG __RPC_FAR* pcbRead ); STDMETHODIMP Write( const void __RPC_FAR* pv, ULONG cb, ULONG __RPC_FAR* pcbWritten ); void ResetPosition( ) { m_uPosition = 0U; }; HRESULT PreAllocBuffer( ULONG uSize ); private: ULONG m_uRefCount; // reference count void* m_pBuffer; // buffer ULONG m_uBufferUsed; // buffer used ULONG m_uBufferSize; // buffer size ULONG m_uPosition; // current index position in the buffer }; 实现如下: CSequentialStream::CSequentialStream( ) : m_uRefCount( 0U ), m_pBuffer( NULL ), m_uBufferUsed( 0U ), m_uBufferSize( 0U ), m_uPosition( 0U ) { AddRef( ); } CSequentialStream::~CSequentialStream( ) { Clear( ); } HRESULT CSequentialStream::QueryInterface( REFIID riid, void** ppv ) { if( riid == IID_IUnknown || riid == IID_ISequentialStream ) { * ppv = this; ( ( IUnknown* )*ppv )->AddRef( ); return S_OK; } * ppv = NULL; return E_NOINTERFACE; } BOOL CSequentialStream::Seek( ULONG uPosition ) { ASSERT( uPosition < m_uBufferUsed ); m_uPosition = uPosition; return TRUE; } BOOL CSequentialStream::Clear( ) { m_uBufferUsed = 0U; m_uBufferSize = 0U; m_uPosition = 0U; ( m_pBuffer != NULL ? CoTaskMemFree( m_pBuffer ) : 0 ); m_pBuffer = NULL; return TRUE; } HRESULT CSequentialStream::PreAllocBuffer( ULONG uSize ) { if( m_uBufferSize < uSize ) { m_uBufferSize = uSize; m_pBuffer = CoTaskMemRealloc( m_pBuffer, m_uBufferSize ); if( m_pBuffer == NULL ) { Clear( ); return STG_E_INSUFFICIENTMEMORY; } } return S_OK; } HRESULT CSequentialStream::Read( void* pv, ULONG cb, ULONG* pcbRead ) { ( pcbRead != NULL ? ( * pcbRead = 0U ) : 0 ); if( pv == NULL ) { return STG_E_INVALIDPOINTER; } if( cb == 0U ) { return S_OK; } ASSERT( m_uPosition uBytesLeft ? uBytesLeft : cb ); memcpy( pv, ( BYTE* )m_pBuffer + m_uPosition, uBytesRead ); m_uPosition += uBytesRead; ( pcbRead != NULL ? ( * pcbRead = uBytesRead ) : 0 ); return ( cb != uBytesRead ? S_FALSE : S_OK ); } HRESULT CSequentialStream::Write( const void* pv, ULONG cb, ULONG* pcbWritten ) { if( pv == NULL ) { return STG_E_INVALIDPOINTER; } ( pcbWritten != NULL ? ( * pcbWritten = 0U ) : 0 ); if( cb == 0U ){ return S_OK; } ASSERT( m_uPosition QueryInterface( IID_ICommandProperties, ( void** )&pICommandProperties ); // 设置 Rowset 属性为“可以更新某字段的值” hResult = pICommandProperties->SetProperties( 1, &dbPropSet ); hResult = pICommandText->SetCommandText( DBGUID_DBSQL, ( LPCOLESTR )L"SELECT Contents FROM News WHERE ID = @@IDENTITY" ); LONG lAffected; EAutoReleasePtr pIRowsetChange; hResult = pICommand->Execute( NULL, IID_IRowsetChange, NULL, &lAffected, ( IUnknown** )&pIRowsetChange ); EAutoReleasePtr pIAccessor; hResult = pIRowsetChange->QueryInterface( IID_IAccessor, ( void** )&pIAccessor ); struct BLOBDATA { DBSTATUS dwStatus; DWORD dwLength; ISequentialStream* pISeqStream; }; // 有关DBOBJECT、DBBINDING的设置,建议参考MSDN,很容易懂。 DBOBJECT dbObj; dbObj.dwFlags = STGM_READ; dbObj.iid = IID_ISequentialStream; DBBINDING dbBinding; dbBinding.iOrdinal = 1; // BLOB 字段的位置,从 1 开始 dbBinding.obValue = offsetof( BLOBDATA, pISeqStream ); dbBinding.obLength = offsetof( BLOBDATA, dwLength ); dbBinding.obStatus = offsetof( BLOBDATA, dwStatus ); dbBinding.pTypeInfo = NULL; dbBinding.pObject = &dbObj; dbBinding.pBindExt = NULL; dbBinding.dwPart = DBPART_VALUE | DBPART_STATUS | DBPART_LENGTH; dbBinding.dwMemOwner = DBMEMOWNER_CLIENTOWNED; dbBinding.eParamIO = DBPARAMIO_NOTPARAM; dbBinding.cbMaxLen = 0; dbBinding.dwFlags = 0; dbBinding.wType = DBTYPE_IUNKNOWN; dbBinding.bPrecision = 0; dbBinding.bScale = 0; HACCESSOR hAccessor = DB_NULL_HACCESSOR; DBBINDSTATUS dbs; hResult = pIAccessor->CreateAccessor( DBACCESSOR_ROWDATA, 1, &dbBinding, sizeof( BLOBDATA ), &hAccessor, &dbs ); EAutoReleasePtr pIRowset; hResult = pIRowsetChange->QueryInterface( IID_IRowset, ( void** )&pIRowset ); ULONG uRowsObtained = 0; HROW* phRows = NULL; hResult = pIRowset->GetNextRows( NULL, 0, 1, &uRowsObtained, &phRows ); CSequentialStream* pss = new CSequentialStream; pss->PreAllocBuffer( 1024 ); // 预先分配好内存,并读入数据 pss->Write( pszSomebuffer, 512, NULL ); // pss->Write可以连续调用 pss->Write( pszSomebuffer+512, 512, NULL ); pss->ResetPosition( ); BLOBDATA bd; bd.pISeqStream = ( ISequentialStream* )pss; bd.dwStatus = DBSTATUS_S_OK; bd.dwLength = pss->GetLength( ); // 将 BLOB 数据写入到数据库 hResult = pIRowsetChange->SetData( phRows[0], hAccessor, &bd ); pIAccessor->ReleaseAccessor( hAccessor, NULL ); pIRowset->ReleaseRows( uRowsObtained, phRows, NULL, NULL, NULL ); // pss was released by pIRowsetChange->SetData. 这样,我们就完成了一条记录的添加。读取BLOB字段的代码跟上面的完全类似,只要把 hResult = pIRowset->GetNextRows( NULL, 0, 1, &uRowsObtained, &phRows ); 后面的那些改成下面的代码即可。 BLOBDATA bd; hResult = pIRowset->GetData( phRows[0], hAccessor, &bd ); if( bd.dwStatus == DBSTATUS_S_ISNULL ) { // 此字段为空 } else if( bd.dwStatus != DBSTATUS_S_OK || bd.pISeqStream == NULL ) { // 失败 } else { // 从系统分配的 ISequentialStream 接口读入 BLOB 数据 BYTE szReadBuffer[1024]; for( ULONG uRead = 0U; ; ) { if( FAILED( bd.pISeqStream->Read( szReadBuffer, 1024, &uRead ) ) ) { break; } //szReadBuffer中就包含了BLOB字段的数据 if( uRead != 1024 ) { break; } } bd.pISeqStream->Release( ); } pIAccessor->ReleaseAccessor( hAccessor, NULL ); pIRowset->ReleaseRows( uRowsObtained, phRows, NULL, NULL, NULL ); 5. OLE DB Templates连接 使用OLE DB接口编程属于最低可能层,代码冗长并且很难维护。因此MS Visual Studio对OLE DB进一步抽象和封装,提供COM OLE DB Templates这个可行的中间层,从而简化了OLE DB应用程序的编写。 OLE DB Templates编写客户数据库程序方法: 以MFC AppWizard为向导建立应用程序框架,添加OLE DB支持的头文件,然后使用OLE DB类进行数据库应用开发。 以ATL COM AppWizard为向导建立应用程序框架,该框架直接支持OLE DB模板类。 OLE DB Templates包括:Consumer Templates和Provider Templates。 (1) Consumer Templates使用者模板 使用者模板(Consumer Templates)体系结构: (2) Provider Templates服务器模板 服务器模板类体系结构: 6. ADO连接 ADO(ActiveX Data Object,ActiveX数据对象)是MS为最新和最强大的数据访问接口OLE DB而设计,是一个便于使用的应用程序层接口。ADO是一种面向对象的、与语言无关的(Language_Neutral)数据访问应用编程接口。它对 OLE DB API进行封装,实现对数据的高层访问,同时它也提供了多语言的访问技术,此外,由于ADO提供了访问自动化接口,它也支持脚本语言。ADO最主要的优点在于易于使用、速度快、内存支出少和磁盘遗迹小。ADO是用来访问OLE DB的数据库技术。在模型层次上它基于OLE DB,但在应用上又高于OLE DB,因此它简化了对对象模型的操作,并且不依赖于对象之间的相互层次关系。但是OLE的接口可以数据提供程序、服务提供程序和数据使用程序使用,而 ADO所提供的对象只能被数据应用程序使用。并且,ADO对象使用了OLE DB服务提供程序和OLE DB数据提供程序所提供的接口和服务。 (1)ADO访问数据库的结构原理图: (2)ADO对象模型: ADO对象模型包括以下关键对象: Connection对象:在数据库应用里操作数据源都通过该对象,这是数据交换的环境,代表与数据源的一个会话。 Command对象:是一个对数据源执行命令的定义。 Parameter对象:用于制定参数化查询或者存储过程的参数。 Recordset对象:是执行结果集存储到本地的ADO对象。 Field对象:ADO中对列进行操作的对象。 Error对象:对ADO数据操作时发生错误的详细描述。 Property对象:代表一个由提供者定义的ADO对象的动态特征。 ADO对象编程模型: Parameters Collection Execute Source Error Collection (Optional) Active Fields Connection Collection (3)ADO编程一般步骤: 创建一个Connection对象。 打开数据源,建立同数据源的连接。 执行一个SQL命令。 使用结果集。 终止连接。 (4)ADO的数据库访问规范 引入ADO支持 #import Program Files\Common Files\System\ado\msado*.dll 初始化和释放ADO环境: CoInitialize(NULL); CoUninitialize(); 封装的ADO类A set of ADO classes: http://blog.csdn.net/byxdaz/archive/2008/06/19/2563174.aspx ADO编程小结:http://hi.baidu.com/sunkanghome/blog/item/273171f9ffb4735c252df286.html http://hi.baidu.com/sunkanghome/blog/item/cea70101bdb177031d95839a.html 应用实例: 在Visual C++中用ADO进行数据库编程 1. 生成应用程序框架并初始化OLE/COM库环境   创建一个标准的MFC AppWizard(exe)应用程序,然后在使用ADO数据库的InitInstance函数中初始化OLE/COM库(因为ADO库是一个COM DLL库)。 本例为:  BOOL CAdotestDlg::OnInitDialog()  { ::CoInitialize(NULL); //初始化OLE/COM库环境 } 程序最后要调用 ::CoUninitialize();//释放程序占用的COM 资源。 另外: m_pRecordset->Close(); 注意!!!不要多次关闭!!!!!!!!!!!! m_pConnection->Close(); m_pRecordset = NULL; m_pConnection = NULL;  2. 引入ADO库文件   使用ADO前必须在工程的stdafx.h文件最后用直接引入符号#import引入ADO库文件,以使编译器能正确编译。代码如下: #import "C:\Program Files\common files\system\ado\msado15.dll" no_namespace rename("EOF","adoEOF")   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。   class CAdotestDlg : public CDialog { _ConnectionPtr m_pConnection; _RecordsetPtr m_pRecordset;    ClistCtrl m_List; ...... } ADO库包含三个智能指针:_ConnectionPtr、_CommandPtr和_RecordsetPtr。 _ConnectionPtr通常被用来创建一个数据连接或执行一条不返回任何结果的SQL语句,如一个存储过程。 _CommandPtr返回一个记录集。它提供了一种简单的方法来执行返回记录集的存储过程和SQL语句。在使用_CommandPtr接口时,可以利用全局_ConnectionPtr接口,也可以在_CommandPtr接口里直接使用连接串。_RecordsetPtr是一个记录集对象。与以上两种对象相比,它对记录集提供了更多的控制功能,如记录锁定、游标控制等。   在使用ADO程序的事件响应中OnButton1加入以下代码:   void CAdotestDlg::OnButton1() { m_List.ResetContent(); m_pConnection.CreateInstance(_uuidof(Connection)); //初始化Connection指针 m_pRecordset.CreateInstance(_uuidof(Recordset));// 初始化Recordset指针 try { m_pConnection->Open("DSN=ADOTest","","",0); //连接叫作ADOTest的ODBC数据源 //注意:这是连接不需要用户ID或密码的open 函数 // 否则形式为 ->Open("DSN=test;uid=sa;pwd=123;","","",0); // 执行SQL语句得到一个记录集把其指针赋值给m_pRecordset CString strSql="select * from middle"; BSTR bstrSQL = strSql.AllocSysString(); m_pRecordset->Open(bstrSQL,(IDispatch*)m_pConnection,adOpenDynamic,adLockOptimistic,adCmdText); //adOpenDynamic:动态 adLockOptimistic乐观封锁法 adCmdText:文本查询语句 while(!m_pRecordset->adoEOF)// 遍历所有记录 { //取纪录字段值方式之一 _variant_t TheValue; //VARIANT数据类型 TheValue = m_pRecordset->GetCollect("BIG_NAME");//得到字段BIG_NAME的值 if(TheValue.vt!=VT_NULL) m_List.AddString((char*)_bstr_t(TheValue)); //将该值加入到列表控件中 //取纪录字段值方式之二 // _bstr_t TheValue1=m_pRecordset->Fields->GetItem("BIG_NAME")->Value; // CString temp=TheValue1.copy(); // m_List.AddString(temp); //数据类型转换 _variant_t vUsername,vBirthday,vID,vOld; TRACE("id:%d,姓名:%s,年龄:%d,生日:%s\r\n", vID.lVal,(LPCTSTR)(_bstr_t)vUsername,vOld.lVal,(LPCTSTR)(_bstr_t)vBirthday); m_pRecordset->MoveNext();//转到下一条纪录 } m_pRecordset->Close(); m_pConnection->Close(); } catch (_com_error e)//异常处理 { AfxMessageBox(e.ErrorMessage()); } m_pRecordset->Close(); //注意!!!不要多次关闭!!!!否则会出错 m_pConnection->Close(); m_pRecordset = NULL; m_pConnection = NULL; }   程序中通过_variant_t和_bstr_t转换 COM对象和C++类型的数据, _variant_t类封装了OLE自治VARIANT数据类型。在C++中使用_variant_t类要比直接使用VARIANT数据类型容易得多。   好,编译后该程序就能运行了,但记住运行前要创建一个叫ADOTest的ODBC数据源。该程序将把表middle中的BIG_NAME字段值显示在列表控件中。  4.执行SQL命令并取得结果记录集 为了取得结果记录集,我们定义一个指向Recordset对象的指针:_RecordsetPtr m_pRecordset; 并为其创建 Recordset对象的实例: m_pRecordset.CreateInstance("ADODB.Recordset"); SQL命令的执行可以采用多种形式,下面我们一进行阐述。 (1)利用Connection对象的Execute方法执行SQL命令 Execute 方法的原型如下所示: _RecordsetPtr Connection15::Execute ( _bstr_t CommandText, VARIANT * RecordsAffected, long Options ) 其中CommandText是命令字串,通常是SQL命令。 参数RecordsAffected是操作完成后所影响的行数, 参数Options表示CommandText中内容的类型,Options可以取如下值之一: adCmdText:表明CommandText是文本命令 adCmdTable:表明CommandText是一个表名 adCmdProc:表明CommandText是一个存储过程 adCmdUnknown: 未知 Execute执行完后返回一个指向记录集的指针,下面我们给出具体代码并作说明。 _variant_t RecordsAffected; ///执行SQL命令:CREATE TABLE创建表格users,users包含四个字段:整形ID,字符串username,整形old,日期型birthday m_pConnection->Execute("CREATE TABLE users(ID INTEGER,username TEXT,old INTEGER,birthday DATETIME)", &RecordsAffected, adCmdText); ///往表格里面添加记录 m_pConnection->Execute("INSERT INTO users(ID,username,old,birthday) VALUES (1, 'Washington',25,'1970/1/1')",&RecordsAffected,adCmdText); ///将所有记录old字段的值加一 m_pConnection->Execute("UPDATE users SET old = old+1",&RecordsAffected,adCmdText); ///执行SQL统计命令得到包含记录条数的记录集 m_pRecordset = m_pConnection->Execute("SELECT COUNT(*) FROM users",&RecordsAffected,adCmdText); _variant_t vIndex = (long)0; _variant_t vCount = m_pRecordset->GetCollect(vIndex);///取得第一个字段的值放入vCount变量 上两句可以写成— _variant_t vCount = m_pRecordset->GetCollect((_variant_t)((long)0)); m_pRecordset->Close();///关闭记录集 CString message; message.Format("共有%d条记录",vCount.lVal); AfxMessageBox(message);///显示当前记录条数 (2)利用Command对象来执行SQL命令 _CommandPtr m_pCommand; m_pCommand.CreateInstance("ADODB.Command"); _variant_t vNULL; vNULL.vt = VT_ERROR; vNULL.scode = DISP_E_PARAMNOTFOUND;///定义为无参数 m_pCommand->ActiveConnection = m_pConnection;///非常关键的一句,将建立的连接赋值给它 m_pCommand->CommandText = "SELECT * FROM users";///命令字串 m_pRecordset = m_pCommand->Execute(&vNULL,&vNULL,adCmdText);///执行命令,取得记录集 在这段代码中我们只是用Command对象来执行了SELECT查询语句,Command对象在进行存储过程的调用中能真正体现它的作用。下次我们将详细介绍。 (3)直接用Recordset对象进行查询取得记录集 实例—— void CGmsaDlg::OnDBSelect() { // TODO: Add your control notification handler code here _RecordsetPtr Rs1; //定义Recordset对象 _bstr_t Connect("DSN=GMS;UID=sa;PWD=;");//定义连接字符串 _bstr_t Source ("SELECT count(*) FROM buaa.mdb010"); //要执行的SQL语句 ::CoInitialize(NULL); //初始化Rs1对象 HRESUL hr = Rs1.CreateInstance( __uuidof( Recordset ) ); //省略对返回值hr的判断 Rs1->Open( Source, Connect, adOpenForwardOnly, adLockReadOnly, -1 ); _variant_t temp=Rs1->GetCollect(_variant_t((long)0)); CString strTemp=(char* )(_bstr_t)temp; MessageBox("OK!"+strTemp); } 例如 m_pRecordset->Open("SELECT * FROM users", _variant_t((IDispatch *)m_pConnection,true), adOpenStatic, adLockOptimistic, adCmdText); Open 方法的原型是这样的: HRESULT Recordset15::Open ( const _variant_t & Source, const _variant_t & ActiveConnection, enum CursorTypeEnum CursorType, enum LockTypeEnum LockType, long Options ) 其中: ①Source是数据查询字符串 ②ActiveConnection是已经建立好的连接(我们需要用Connection对象指针来构造一个_variant_t对象) ③CursorType光标类型,它可以是以下值之一,请看这个枚举结构: enum CursorTypeEnum { adOpenUnspecified = -1,///不作特别指定 adOpenForwardOnly = 0,///前滚静态光标。这种光标只能向前浏览记录集,比如用MoveNext向前滚动,这种方式可以提高浏览速度。但诸如 BookMark,RecordCount,AbsolutePosition,AbsolutePage都不能使用 adOpenKeyset = 1,///采用这种光标的记录集看不到其它用户的新增、删除操作,但对于更新原有记录的操作对你是可见的。 adOpenDynamic = 2,///动态光标。所有数据库的操作都会立即在各用户记录集上反应出来。 adOpenStatic = 3///静态光标。它为你的记录集产生一个静态备份,但其它用户的新增、删除、更新操作对你的记录集来说是不可见的。 }; ④LockType 锁定类型,它可以是以下值之一,请看如下枚举结构: enum LockTypeEnum { adLockUnspecified = -1,///未指定 adLockReadOnly = 1,///只读记录集 adLockPessimistic = 2,悲观锁定方式。数据在更新时锁定其它所有动作,这是最安全的锁定机制 adLockOptimistic = 3,乐观锁定方式。只有在你调用Update方法时才锁定记录。在此之前仍然可以做数据的更新、插入、删除等动作 adLockBatchOptimistic = 4,乐观分批更新。编辑时记录不会锁定,更改、插入及删除是在批处理模式下完成。 }; ⑤Options可以取如下值之一: adCmdText: 表明CommandText是文本命令 adCmdTable:表明CommandText是一个表名 adCmdProc:表明 CommandText是一个存储过程 adCmdUnknown:未知 5. 记录集的遍历、更新 根据我们刚才通过执行SQL命令建立好的users表,它包含四个字段:ID,username,old,birthday 以下的代码实现:打开记录集,遍历所有记录,删除第一条记录,添加三条记录,移动光标到第二条记录, 更改其年龄,保存到数据库。 _variant_t vUsername,vBirthday,vID,vOld; _RecordsetPtr m_pRecordset; m_pRecordset.CreateInstance("ADODB.Recordset"); m_pRecordset->Open("SELECT * FROM users", _variant_t((IDispatch*)m_pConnection,true), adOpenStatic, adLockOptimistic, adCmdText); while(!m_pRecordset->adoEOF) { vID = m_pRecordset->GetCollect(_variant_t((long)0));///取得第1列的值,从0开始计数, ///你也可以直接给出列的名称,如下一行 vUsername = m_pRecordset->GetCollect("username");///取得username字段的值 vOld = m_pRecordset->GetCollect("old"); vBirthday = m_pRecordset->GetCollect("birthday"); ///在DEBUG方式下的OUTPUT窗口输出记录集中的记录 if(vID.vt != VT_NULL && vUsername.vt != VT_NULL && vOld.vt != VT_NULL && vBirthday.vt != VT_NULL) TRACE("id:%d,姓名:%s,年龄:%d,生日:%s\r\n", vID.lVal, (LPCTSTR)(_bstr_t)vUsername, vOld.lVal, (LPCTSTR)(_bstr_t)vBirthday); m_pRecordset->MoveNext();///移到下一条记录 } m_pRecordset->MoveFirst();/// 移到首条记录 m_pRecordset->Delete(adAffectCurrent);///删除当前记录 ///添加三条新记录并赋值 for(int i=0;iAddNew();///添加新记录 m_pRecordset->PutCollect("ID",_variant_t((long)(i+10))); m_pRecordset->PutCollect("username",_variant_t("叶利钦")); m_pRecordset->PutCollect("old",_variant_t((long)71)); m_pRecordset->PutCollect("birthday",_variant_t("1930-3-15")); } m_pRecordset->Move(1,_variant_t((long)adBookmarkFirst));/// 从第一条记录往下移动一条记录,即移动到第二条记录处 m_pRecordset->PutCollect(_variant_t("old"),_variant_t((long)45));/// 修改其年龄 m_pRecordset->Update();///保存到库中 备注:多次查询可把查询过程做成一个函数ExecuteSQL让m_pRecordset获得连接指针m_pConnection查询结果 void ExecuteSQL(_ConnectionPtr m_pConnection, _RecordsetPtr m_pRecordset,CString strSql) { //执行Select 语句 BSTR bstrSQL = strSql.AllocSysString(); try { m_pRecordset->Open(bstrSQL,(IDispatch*)m_pConnection,adOpenDynamic,adLockOptimistic,adCmdText); //adOpenDynamic:动态 adLockOptimistic乐观封锁法 adCmdText:文本查询语句 } catch(_com_error error) { CString errorMessage; errorMessage.Format("%s",(LPTSTR)error.Description()); AfxMessageBox(errorMessage); } } //出错处理: 3127 ——没有找到目标表 3092——目标表已经存在 例如: catch(const _com_error e) { AfxMessageBox(e.Description()); long errorCode=e.WCode(); if(3127==errorCode) AfxMessageBox("表不存在"); if(3092==errorCode) AfxMessageBox("表已经存在"); return FALSE; } 7、ADO.NET ADO.NET是一组用于和数据源进行交互的面向对象类库。 ADO.NET的主要对象有哪些? Connection :用于连接到数据库和管理对数据库的事务; Command :用于对数据库发出SQL命令; DataReader :用于从数据源读取只进数据记录流; DataSet :用于对单层数据、XML数据和关系数据进行存储、远程处理和编程; DataAdapter :用于将数据推入DataSet,并使数据与数据库保持一致; ADO.NET 2.0 快速入门: http://tech.e800.com.cn/articles/2009/721/1248143977078_1.html 用VC轻松实现 ADO.net: http://www.vckbase.com/document/viewdoc/?id=1714 Oracle专用方法 1. OCI(Oracle Call Interface)访问 OCI(Oracle Call Interface)是由Oracle提供的一系列用于访问Oracles数据库服 务器的标准接口,它可以使用户将Oracle调用直接嵌入到高级语言中。 使用OCI应用程序访问数据库原理: 在高级语言中使用OCI编程的原理图: 用OCI开发Oracle客户端软件的一般流程: 初始化OCI编程环境 分配必要的句柄,建立服务器连接和一个用户会话 向服务器发出请求,进行必要的数据处理 释放不再需要的语句和句柄 终止会话和连接 2. Oracle Object OLE C++ Class Library 这个类库是一个提供编程接口访问Oracle对象服务器的C++类库,它是用OLE的方式实现的。Oracle提供的是一个进程内服务器,也就是服务器将与应用程序在同一个地址空间内, 它以DLL方式提供。应用程序在访问数据库之前必须先加载Oracle对象服务器(OStatup方法),然后与Oracle对象服务器通信,Oracle对象服务器其实是一些组件,它通过Oracle的OCI访问数据库。 Oracle对象服务器其实是一些COM组件,它通过Oracle的OCI访问数据库。 运用Oracle Objects for OLE C++ Class Library开发的步骤: 1>通过调用OStatup方法初始化类库。 2>连接数据库。 3>操纵数据库 断开数据库(类库自动为你自动执行) 4>通过调用OShutdown方法卸载类库。 使用OTL进行数据库编程 OTL 是 Oracle, Odbc and DB2-CLI Template Library 的缩写,是一个C++编译中操控关系数据库的模板库,它目前几乎支持所有的当前各种主流数据库,例如Oracle, MS SQL Server, Sybase, Informix, MySQL, DB2, Interbase / Firebird, PostgreSQL, SQLite, SAP/DB, TimesTen, MS ACCESS等等。OTL中直接操作Oracle主要是通过Oracle提供的OCI接口进行,进行操作DB2数据库则是通过CLI接口来进行,至于MS的数据库和其它一些数据库,则OTL只提供了ODBC来操作的方式。当然Oracle和DB2也可以由OTL间接使用ODBC的方式来进行操纵。 在MS Windows and Unix 平台下,OTL目前支持的数据库版本主要有:Oracle 7 (直接使用 OCI7), Oracle 8 (直接使用 OCI8), Oracle 8i (直接使用OCI8i), Oracle 9i (直接使用OCI9i), Oracle 10g (直接使用OCI10g), DB2 (直接使用DB2 CLI), ODBC 3.x ,ODBC 2.5。OTL最新版本为4.0,参见http://otl.sourceforge.net/,下载地址http://otl.sourceforge.net/otlv4_h.zip。 优点: a. 跨平台 b. 运行效率高,与C语言直接调用API相当 c. 开发效率高,起码比ADO.net使用起来更简单,更简洁 d. 部署容易,不需要ADO组件,不需要.net framework 等 缺点: a. 说明文档以及范例不足够丰富(暂时性的) 其实现在它提供有377个使用范例可参考,下载地址:http://otl.sourceforge.net/otl4_examples.zip。 建立数据源 1.依次点击“开始->控制面板”,打开“控制面板”界面,双击“管理工具”,然后再双击“数据源(ODBC)”,就打开了“ODBC数据源管理器”,选择“系统DSN”。 2.单击“添加”,弹出“创建新数据源”对话框,选择“Microsoft Access Driver(*.mdb)”。 3.点击“完成”,弹出“ODBC Microsoft Access安装”对话框,单击“创建”,开始创建数据库,弹出“新建数据库”对话框,添加数据库名称my_db和选择数据库存放目录,单击“确定”,创建完成,然后添加数据源名:my_db。点击“确定”。 4.然后在系统数据源中就有我们刚才添加的数据源。 5.单击“确定”,完成数据源的创建。 OTL编程 下面我们用一个实例来说明: 1. 创建数据表:TestTable ( ColumA int , ColumB varchar(50),ColumC varchar(50) ) 2. 插入100条数据,ColumA 为数据的 id 范围:0-99 , ColumB=”Test Data %d” , 其中 %d=id 。 3. 删除表中ColumA 中小于10和大于90的数据。 4. 将ColumA为3的倍数的记录中ColumC更新为ColumB的内容。 具体代码为: #include using namespace std; #include #include #include #define OTL_ODBC // 编译 OTL 4.0/ODBC // #define OTL_ODBC_UNIX // 如果在Unix下使用UnixODBC,则需要这个宏 #include "otlv4.h" // 包含 OTL 4.0 头文件 otl_connect db; // 连接对象 //此函数完成插入100条数据,ComulA为数据的id,范围为0-99, //ColumB="Test Data %d",其中%d=id void insert() // 向表中插入行 { // 打开一个通用的流,以模板的方式向表中插入多项数据 otl_stream o(1, // 流的缓冲值必须设置为1 "insert into TestTable values(:f1,:f2,:f3)", // SQL 语句 db // 连接对象 ); char tmp1[32]; char tmp2[30]; for(int i=0;i

阅读(4915) | 评论(0)


版权声明:编程爱好者网站为此博客服务提供商,如本文牵涉到版权问题,编程爱好者网站不承担相关责任,如有版权问题请直接与本文作者联系解决。谢谢!

评论

暂无评论
您需要登录后才能评论,请 登录 或者 注册