ADO-2
?
VC開發(fā)數(shù)據(jù)庫基礎(chǔ)之ADO篇
?
一、ADO簡介
ADO(ActiveX Data Object)是Microsoft數(shù)據(jù)庫應(yīng)用程序開發(fā)的新接口,是建立在OLE DB之上的高層數(shù)據(jù)庫訪問技術(shù),請不必為此擔(dān)心,即使你對OLE DB,COM不了解也能輕松對付ADO,因?yàn)樗浅:唵我子?#xff0c;甚至比你以往所接觸的ODBC API、DAO、RDO都要容易使用,并不失靈活性。本文將詳細(xì)地介紹在VC下如何使用ADO來進(jìn)行數(shù)據(jù)庫應(yīng)用程序開發(fā),并給出示例代碼。
本文示例代碼
二、基本流程
萬事開頭難,任何一種新技術(shù)對于初學(xué)者來說最重要的還是“入門”,掌握其要點(diǎn)。讓我們來看看ADO數(shù)據(jù)庫開發(fā)的基本流程吧!
(1)初始化COM庫,引入ADO庫定義文件
(2)用Connection對象連接數(shù)據(jù)庫
(3)利用建立好的連接,通過Connection、Command對象執(zhí)行SQL命令,或利用Recordset對象取得結(jié)果記錄集進(jìn)行查詢、處理。
(4)使用完畢后關(guān)閉連接釋放對象。
準(zhǔn)備工作:
為了大家都能測試本文提供的例子,我們采用Access數(shù)據(jù)庫,您也可以直接在我們提供的示例代碼中找到這個test.mdb。
下面我們將詳細(xì)介紹上述步驟并給出相關(guān)代碼。
【1】COM庫的初始化
我們可以使用AfxOleInit()來初始化COM庫,這項(xiàng)工作通常在CWinApp::InitInstance()的重載函數(shù)中完成,請看如下代碼:
BOOL CADOTest1App::InitInstance()
{
AfxOleInit();
......
【2】用#import指令引入ADO類型庫
我們在stdafx.h中加入如下語句:(stdafx.h這個文件哪里可以找到?你可以在FileView中的Header Files里找到)
#import "c:/program files/common files/system/ado/msado15.dll" no_namespace rename("EOF","adoEOF")
這一語句有何作用呢?其最終作用同我們熟悉的#include類似,編譯的時候系統(tǒng)會為我們生成msado15.tlh,ado15.tli兩個C++頭文件來定義ADO庫。
幾點(diǎn)說明:
(1) 您的環(huán)境中msado15.dll不一定在這個目錄下,請按實(shí)際情況修改
(2) 在編譯的時候肯能會出現(xiàn)如下警告,對此微軟在MSDN中作了說明,并建議我們不要理會這個警告。
msado15.tlh(405) : warning C4146: unary minus operator applied to unsigned type, result still unsigned
【3】創(chuàng)建Connection對象并連接數(shù)據(jù)庫
首先我們需要添加一個指向Connection對象的指針:
_ConnectionPtr m_pConnection;
下面的代碼演示了如何創(chuàng)建Connection對象實(shí)例及如何連接數(shù)據(jù)庫并進(jìn)行異常捕捉。
BOOL CADOTest1Dlg::OnInitDialog()
{
CDialog::OnInitDialog();
HRESULT hr;
try
{
hr = m_pConnection.CreateInstance("ADODB.Connection");///創(chuàng)建Connection對象
if(SUCCEEDED(hr))
{
hr = m_pConnection->Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=test.mdb","","",adModeUnknown);///連接數(shù)據(jù)庫
///上面一句中連接字串中的Provider是針對ACCESS2000環(huán)境的,對于ACCESS97,需要改為:Provider=Microsoft.Jet.OLEDB.3.51; }
}
catch(_com_error e)///捕捉異常
{
CString errormessage;
errormessage.Format("連接數(shù)據(jù)庫失敗!/r/n錯誤信息:%s",e.ErrorMessage());
AfxMessageBox(errormessage);///顯示錯誤信息
}
在這段代碼中我們是通過Connection對象的Open方法來進(jìn)行連接數(shù)據(jù)庫的,下面是該方法的原型
HRESULT Connection15::Open ( _bstr_t ConnectionString, _bstr_t UserID, _bstr_t Password, long Options )
ConnectionString為連接字串,UserID是用戶名, Password是登陸密碼,Options是連接選項(xiàng),用于指定Connection對象對數(shù)據(jù)的更新許可權(quán),
Options可以是如下幾個常量:
adModeUnknown:缺省。當(dāng)前的許可權(quán)未設(shè)置
adModeRead:只讀
adModeWrite:只寫
adModeReadWrite:可以讀寫
adModeShareDenyRead:阻止其它Connection對象以讀權(quán)限打開連接
adModeShareDenyWrite:阻止其它Connection對象以寫權(quán)限打開連接
adModeShareExclusive:阻止其它Connection對象打開連接
adModeShareDenyNone:允許其它程序或?qū)ο笠匀魏螜?quán)限建立連接
我們給出一些常用的連接方式供大家參考:
(1)通過JET數(shù)據(jù)庫引擎對ACCESS2000數(shù)據(jù)庫的連接
m_pConnection->Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=C://test.mdb","","",adModeUnknown);
(2)通過DSN數(shù)據(jù)源對任何支持ODBC的數(shù)據(jù)庫進(jìn)行連接:
m_pConnection->Open("Data Source=adotest;UID=sa;PWD=;","","",adModeUnknown);
(3)不通過DSN對SQL SERVER數(shù)據(jù)庫進(jìn)行連接: m_pConnection->Open("driver={SQL Server};Server=127.0.0.1;DATABASE=vckbase;UID=sa;PWD=139","","",adModeUnknown);
其中Server是SQL服務(wù)器的名稱,DATABASE是庫的名稱
Connection對象除Open方法外還有許多方法,我們先介紹Connection對象中兩個有用的屬性ConnectionTimeOut與State
ConnectionTimeOut用來設(shè)置連接的超時時間,需要在Open之前調(diào)用,例如: m_pConnection->ConnectionTimeout = 5;///設(shè)置超時時間為5秒
m_pConnection->Open("Data Source=adotest;","","",adModeUnknown);
State屬性指明當(dāng)前Connection對象的狀態(tài),0表示關(guān)閉,1表示已經(jīng)打開,我們可以通過讀取這個屬性來作相應(yīng)的處理,例如:
if(m_pConnection->State)
m_pConnection->Close(); ///如果已經(jīng)打開了連接則關(guān)閉它
【4】執(zhí)行SQL命令并取得結(jié)果記錄集
為了取得結(jié)果記錄集,我們定義一個指向Recordset對象的指針:_RecordsetPtr m_pRecordset;
并為其創(chuàng)建Recordset對象的實(shí)例: m_pRecordset.CreateInstance("ADODB.Recordset");
SQL命令的執(zhí)行可以采用多種形式,下面我們一進(jìn)行闡述。
(1)利用Connection對象的Execute方法執(zhí)行SQL命令
Execute方法的原型如下所示:
_RecordsetPtr Connection15::Execute ( _bstr_t CommandText, VARIANT * RecordsAffected, long Options ) 其中CommandText是命令字串,通常是SQL命令。參數(shù)RecordsAffected是操作完成后所影響的行數(shù), 參數(shù)Options表示CommandText中內(nèi)容的類型,Options可以取如下值之一:
adCmdText:表明CommandText是文本命令
adCmdTable:表明CommandText是一個表名
adCmdProc:表明CommandText是一個存儲過程
adCmdUnknown:未知
Execute執(zhí)行完后返回一個指向記錄集的指針,下面我們給出具體代碼并作說明。 _variant_t RecordsAffected;
///執(zhí)行SQL命令:CREATE TABLE創(chuàng)建表格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);
///執(zhí)行SQL統(tǒng)計(jì)命令得到包含記錄條數(shù)的記錄集
m_pRecordset = m_pConnection->Execute("SELECT COUNT(*) FROM users",&RecordsAffected,adCmdText);
_variant_t vIndex = (long)0;
_variant_t vCount = m_pRecordset->GetCollect(vIndex);///取得第一個字段的值放入vCount變量
m_pRecordset->Close();///關(guān)閉記錄集
CString message;
message.Format("共有%d條記錄",vCount.lVal);
AfxMessageBox(message);///顯示當(dāng)前記錄條數(shù)
(2)利用Command對象來執(zhí)行SQL命令
_CommandPtr m_pCommand;
m_pCommand.CreateInstance("ADODB.Command");
_variant_t vNULL;
vNULL.vt = VT_ERROR;
vNULL.scode = DISP_E_PARAMNOTFOUND;///定義為無參數(shù)
m_pCommand->ActiveConnection = m_pConnection;///非常關(guān)鍵的一句,將建立的連接賦值給它
m_pCommand->CommandText = "SELECT * FROM users";///命令字串
m_pRecordset = m_pCommand->Execute(&vNULL,&vNULL,adCmdText);///執(zhí)行命令,取得記錄集
在這段代碼中我們只是用Command對象來執(zhí)行了SELECT查詢語句,Command對象在進(jìn)行存儲過程的調(diào)用中能真正體現(xiàn)它的作用。下次我們將詳細(xì)介紹。
(3)直接用Recordset對象進(jìn)行查詢?nèi)〉糜涗浖?
例如
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是數(shù)據(jù)查詢字符串
②ActiveConnection是已經(jīng)建立好的連接(我們需要用Connection對象指針來構(gòu)造一個_variant_t對象)
③CursorType光標(biāo)類型,它可以是以下值之一,請看這個枚舉結(jié)構(gòu):
enum CursorTypeEnum
{
adOpenUnspecified = -1,///不作特別指定
adOpenForwardOnly = 0,///前滾靜態(tài)光標(biāo)。這種光標(biāo)只能向前瀏覽記錄集,比如用MoveNext向前滾動,這種方式可以提高瀏覽速度。但諸如BookMark,RecordCount,AbsolutePosition,AbsolutePage都不能使用
adOpenKeyset = 1,///采用這種光標(biāo)的記錄集看不到其它用戶的新增、刪除操作,但對于更新原有記錄的操作對你是可見的。
adOpenDynamic = 2,///動態(tài)光標(biāo)。所有數(shù)據(jù)庫的操作都會立即在各用戶記錄集上反應(yīng)出來。
adOpenStatic = 3///靜態(tài)光標(biāo)。它為你的記錄集產(chǎn)生一個靜態(tài)備份,但其它用戶的新增、刪除、更新操作對你的記錄集來說是不可見的。
};
④LockType鎖定類型,它可以是以下值之一,請看如下枚舉結(jié)構(gòu):
enum LockTypeEnum
{
adLockUnspecified = -1,///未指定
adLockReadOnly = 1,///只讀記錄集
adLockPessimistic = 2,悲觀鎖定方式。數(shù)據(jù)在更新時鎖定其它所有動作,這是最安全的鎖定機(jī)制
adLockOptimistic = 3,樂觀鎖定方式。只有在你調(diào)用Update方法時才鎖定記錄。在此之前仍然可以做數(shù)據(jù)的更新、插入、刪除等動作
adLockBatchOptimistic = 4,樂觀分批更新。編輯時記錄不會鎖定,更改、插入及刪除是在批處理模式下完成。
};
⑤Options請參考本文中對Connection對象的Execute方法的介紹
【5】記錄集的遍歷、更新
根據(jù)我們剛才通過執(zhí)行SQL命令建立好的users表,它包含四個字段:ID,username,old,birthday
以下的代碼實(shí)現(xiàn):打開記錄集,遍歷所有記錄,刪除第一條記錄,添加三條記錄,移動光標(biāo)到第二條記錄,更改其年齡,保存到數(shù)據(jù)庫。
_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)///這里為什么是adoEOF而不是EOF呢?還記得rename("EOF","adoEOF")這一句嗎?
{
vID = m_pRecordset->GetCollect(_variant_t((long)0));///取得第1列的值,從0開始計(jì)數(shù),你也可以直接給出列的名稱,如下一行
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);///刪除當(dāng)前記錄
///添加三條新記錄并賦值
for(int i=0;i<3;i++)
{
m_pRecordset->AddNew();///添加新記錄
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();///保存到庫中
【6】關(guān)閉記錄集與連接
記錄集或連接都可以用Close方法來關(guān)閉
m_pRecordset->Close();///關(guān)閉記錄集
m_pConnection->Close();///關(guān)閉連接
至此,我想您已經(jīng)熟悉了ADO操作數(shù)據(jù)庫的大致流程,也許您已經(jīng)胸有成竹,也許您還有點(diǎn)胡涂,不要緊!建議你嘗試寫幾個例子,這樣會更好地熟悉ADO,最后我給大家寫了一個小例子,例子中讀出所有記錄放到列表控件中、并可以添加、刪除、修改記錄。
點(diǎn)這里下載示例代碼
后記:限于篇幅ADO中的許多內(nèi)容還沒有介紹,下次我們將詳細(xì)介紹Recordset對象的屬性、方法并解決幾個關(guān)鍵的技術(shù):綁定方式處理記錄集數(shù)據(jù)、存儲過程的調(diào)用、事務(wù)處理、圖象在數(shù)據(jù)庫中的保存與讀取、與表格控件的配合使用等
//--------------------------------------------------------------------------------------
又一個說明:
在vc中怎么使用ADO訪問數(shù)據(jù)庫?
1、#import "msado15.dll" no_namespace rename("EOF","ADOEOF")
? 引入"msado15.dll"動態(tài)庫。
2、定義一個變量? _ConnectionPtr m_AdoCon;代表一個代表與數(shù)據(jù)源進(jìn)行的唯一會話
3、打開數(shù)據(jù)庫
??????? _bstr_t connString;?? //連接字符串
try
{
connString = _T("Provider=MSDASQL.1;Data Source=kong;Persist Security Info=False");
m_AdoCon.CreateInstance(__uuidof(Connection));//???
m_AdoCon->Open(connString, "", "", -1);
}
catch (_com_error& comerr)
{
IErrorInfo* pErrorInfo = comerr.ErrorInfo();
HRESULT hr = comerr.Error();
if (pErrorInfo)
{
BSTR bsDesc = NULL;
pErrorInfo->GetDescription( &bsDesc );
_bstr_t sDesc( bsDesc, false );
AfxMessageBox(sDesc.operator LPCTSTR());
pErrorInfo->Release();
}
?????????? }
4、使用
CString strIp, SQL_STR, strTemp, strResult;
UINT port;
_RecordsetPtr?? AdoRst;?????????????? //RecordSet的實(shí)例
_StreamPtr AdoStream;???????????????? //Ado數(shù)據(jù)流
SQL_STR = m_strText;?????????????????? //SQL語句
if (m_AdoCon->State == adStateClosed) //判斷連接是否已經(jīng)關(guān)閉
{
strTemp = "Ado connection hasn't set up.";
m_pWnd->SendMessage(WM_FRESHWATCHLIST, (WPARAM)(&strTemp));
SendMsg(strTemp, FALSE);
return;
}
try
{
//Create Recordset Interface
AdoRst.CreateInstance(__uuidof(Recordset));
AdoRst->Open((LPCTSTR)SQL_STR, m_AdoCon.GetInterfacePtr(),
adOpenDynamic,adLockOptimistic,adCmdUnknown);?????? //打開并且執(zhí)行SQL(SQL_STR)語句
if (strTemp != "SELECT")
strResult =_T("OK!");
else
{
//Save the xmlResult to strResult
AdoStream.CreateInstance(__uuidof(Stream));??????? //
AdoRst->Save(AdoStream.GetInterfacePtr(), adPersistXML);
strResult = ((BSTR)AdoStream->ReadText(adReadAll));????? //轉(zhuǎn)換為XML格式
}
}
catch (_com_error& comerr)
{
//catch the COM exception
IErrorInfo* pErrorInfo = comerr.ErrorInfo();
HRESULT hr = comerr.Error();
if (pErrorInfo)
{
BSTR bsDesc = NULL;
pErrorInfo->GetDescription( &bsDesc );
_bstr_t sDesc( bsDesc, false);
strResult = sDesc.operator LPCTSTR();
pErrorInfo->Release();
}
}
5.取得表中的字段 GetCollect()
int nItem;
_variant_t vUsername,vBirthday,vID,vOld;
try
{
m_pRecordset.CreateInstance("ADODB.Recordset");
m_pRecordset->Open("SELECT * FROM users",_variant_t((IDispatch*)theApp.m_pConnection,true),adOpenStatic,adLockOptimistic,adCmdText);
m_bSuccess = TRUE;
while(!m_pRecordset->adoEOF)
{
vID = m_pRecordset->GetCollect("ID");
vUsername = m_pRecordset->GetCollect("username");
vOld = m_pRecordset->GetCollect("old");
vBirthday = m_pRecordset->GetCollect("birthday");
nItem=m_userlist.InsertItem(0xffff,(_bstr_t)vID);
m_userlist.SetItem(nItem,1,1,(_bstr_t)vUsername,NULL,0,0,0);
m_userlist.SetItem(nItem,2,1,(_bstr_t)vOld,NULL,0,0,0);
m_userlist.SetItem(nItem,3,1,(_bstr_t)vBirthday,NULL,0,0,0);
m_pRecordset->MoveNext();
}
}
6.添加數(shù)據(jù) PutCollect()
if(!m_pRecordset->adoEOF && m_nCurrentSel >= 0 && m_bAutoSave)
{
vUserID = (long)m_nUserID;
vUsername = m_sUsername;
vOld = (long)m_nOld;
vBirthday = m_tBirthday;
m_pRecordset->PutCollect("ID",vUserID);
m_pRecordset->PutCollect("username",vUsername);
m_pRecordset->PutCollect("old",vOld);
m_pRecordset->PutCollect("birthday",vBirthday)
??????? }
//--------------------------------------------------------------------------------------
在ado中獲取access中表的信息
void OpenSchemaX(TCHAR *TableName)
{
??? HRESULT? hr = S_OK;
IADORecordBinding?? *picRs = NULL;
_RecordsetPtr pRstSchema("ADODB.Recordset");
_ConnectionPtr pConnection("ADODB.Connection" );
pConnection->ConnectionString = TableName;
pConnection->Provider = "Microsoft.Jet.OLEDB.4.0";
?? try
??? {
pConnection->Open(pConnection->ConnectionString, "", "", adModeUnknown);
? pRstSchema->QueryInterface(
????????? __uuidof(IADORecordBinding), (LPVOID*)&picRs);
??????? pRstSchema = pConnection->OpenSchema(adSchemaTables);//枚舉表的名稱處理
??????? while(!(pRstSchema->EndOfFile))
??????? {
CString strTableType;
??????????? _bstr_t table_name = pRstSchema->Fields->
??????????????? GetItem("TABLE_NAME")->Value;//獲取表的名稱
??????????? _bstr_t table_type = pRstSchema->Fields->
??????????????? GetItem("TABLE_TYPE")->Value;//獲取表的類型
??????????? strTableType.Format("%s",(LPCSTR) table_type);
if(!lstrcmp(strTableType,_T("TABLE")))
??????????? {
m_cbTeam.AddString((LPCSTR) table_name);//添加表的名稱
}
??????????? pRstSchema->MoveNext();
??????? }
??????? // Clean up objects before exit.
??????? pRstSchema->Close();
??????? pConnection->Close();
??? }
??? catch (_com_error &e)
??? {
??????? // Notify the user of errors if any.
??????? // Pass a connection pointer accessed from the Connection.???????
??????? PrintProviderError(pConnection);
??????? PrintComError(e);
??? }
}
--------------------------------------------------------------------------------
CSDN VC編程經(jīng)驗(yàn)總結(jié)
?
ADO 2.8 Samples?
See Also
Clear Method | Command Object | Connection Object | Errors Collection | Execute Method (ADO Command) | Execute Method (ADO Connection) | Requery Method
Execute, Requery, and Clear Methods Example (VC++)
This example demonstrates the Execute method when run from both a Command object and a Connection object. It also uses the Requery method to retrieve current data in a recordset, and the Clear method to clear the contents of the Errors collection. The ExecuteCommand and PrintOutput functions are required for this example to run.
// BeginExecuteCpp
#include <ole2.h>
#include <stdio.h>
#import "c:/Program Files/Common Files/System/ADO/msado15.dll" /
??? no_namespace rename("EOF", "EndOfFile")
// Function declarations
inline void TESTHR(HRESULT x) {if FAILED(x) _com_issue_error(x);};
void ExecuteX(void);
void ExecuteCommand(_CommandPtr pCmdTemp, _RecordsetPtr pRstTemp);
void PrintOutput(_RecordsetPtr pRstTemp);
void PrintProviderError(_ConnectionPtr pConnection);
void PrintComError(_com_error &e);
//????? Main Function???????? //
void main()
{
??? if(FAILED(::CoInitialize(NULL)))
??????? return;
??? ExecuteX();
??? ::CoUninitialize();
}
///
//????? ExecuteX Function??????? //
///
void ExecuteX(void)
{
?? HRESULT??? hr = S_OK;
??? // Define string variables.
?? _bstr_t strSQLChange("UPDATE Titles SET Type = "
??????????? "'self_help' WHERE Type = 'psychology'");
?? _bstr_t strSQLRestore("UPDATE Titles SET Type = "
??????????? "'psychology' WHERE Type = 'self_help'");
?? _bstr_t strCnn("Provider='sqloledb';Data Source='MySqlServer';"
??????????? "Initial Catalog='pubs';Integrated Security='SSPI';");
??? // Define ADO object pointers.
??? // Initialize pointers on define.
??? // These are in the ADODB::? namespace.
??? _ConnectionPtr? pConnection = NULL;
??? _CommandPtr???? pCmdChange? = NULL;
??? _RecordsetPtr?? pRstTitles? = NULL;
??? try
??? {
??????? // Open connection.
??????? TESTHR(pConnection.CreateInstance(__uuidof(Connection)));
??????? pConnection->Open (strCnn, "", "", adConnectUnspecified);
??????? // Create command object.
??????? TESTHR(pCmdChange.CreateInstance(__uuidof(Command)));
??????? pCmdChange->ActiveConnection = pConnection;
??????? pCmdChange->CommandText = strSQLChange;
??????? // Open titles table, casting Connection pointer to an
??????? // IDispatch type so converted to correct type of variant.
??????? TESTHR(pRstTitles.CreateInstance(__uuidof(Recordset)));
??????? pRstTitles->Open ("Titles", _variant_t((IDispatch *) pConnection,
??????????? true), adOpenStatic, adLockOptimistic, adCmdTable);
??????? // Print report of original data.
??????? printf(
??????????? "/n/nData in Titles table before executing the query: /n");
??????? // Call function to print loop recordset contents.
??????? PrintOutput(pRstTitles);
??????? // Clear extraneous errors from the Errors collection.
??????? pConnection->Errors->Clear();
??????? // Call ExecuteCommand subroutine to execute pCmdChange command.
??????? ExecuteCommand(pCmdChange, pRstTitles);
??????? // Print report of new data.
??????? printf(
??????????? "/n/n/tData in Titles table after executing the query: /n");
??????? PrintOutput(pRstTitles);
??????? // Use the Connection object's execute method to
??????? // execute SQL statement to restore data.
??????? pConnection->Execute(strSQLRestore, NULL, adExecuteNoRecords);
??????? // Retrieve the current data by requerying the recordset.
??????? pRstTitles->Requery(adCmdUnknown);
??????? // Print report of restored data.
??????? printf(
??????????? "/n/n/tData after exec. query to restore original info: /n");
??????? PrintOutput(pRstTitles);
??? }
??? catch (_com_error &e)
??? {
??????? PrintProviderError(pConnection);
??????? PrintComError(e);
??? }
??? // Clean up objects before exit.
??? if (pRstTitles)
??????? if (pRstTitles->State == adStateOpen)
??????????? pRstTitles->Close();
??? if (pConnection)
??????? if (pConnection->State == adStateOpen)
??????????? pConnection->Close();
}
//
//????? ExecuteCommand Function???????? //
//
void ExecuteCommand(_CommandPtr pCmdTemp, _RecordsetPtr pRstTemp)
{
??? try
??? {
??????? // CommandText property already set before function was called.
??????? pCmdTemp->Execute(NULL, NULL, adCmdText);
??????? // Retrieve the current data by requerying the recordset.
??????? pRstTemp->Requery(adCmdUnknown);
??? }
??? catch(_com_error &e)
??? {
??????? // Notify user of any errors that result from
??????? // executing the query.
??????? // Pass a connection pointer accessed from the Recordset.
??????? PrintProviderError(pRstTemp->GetActiveConnection());
??????? PrintComError(e);
??? }
}
/
//????? PrintOutput Function?????? //
/
void PrintOutput(_RecordsetPtr pRstTemp)
{
??? // Ensure at top of recordset.
??? pRstTemp->MoveFirst();
??? // If EOF is true, then no data and skip print loop.
??? if( pRstTemp->EndOfFile )
??? {
??????? printf("/tRecordset empty/n");
??? }
??? else
??? {
??????? // Define temporary strings for output conversions.
??????? // Initialize to first record's values.
??????? _bstr_t bstrTitle;
??????? _bstr_t bstrType;
??????? // Enumerate Recordset and print from each.
??????? while(!(pRstTemp->EndOfFile))
??????????? {
??????????? // Convert variant string to convertable string type.
??????????? bstrTitle = pRstTemp->Fields->GetItem("Title")->Value;
??????????? bstrType? = pRstTemp->Fields->GetItem("Type")->Value;
??????????? printf("/t%s, %s /n",
??????????????? (LPCSTR) bstrTitle,
??????????????? (LPCSTR) bstrType);
???
??????????? pRstTemp->MoveNext();
??????? }
??? }
}
///
//????? PrintProviderError Function????????? //
///
void PrintProviderError(_ConnectionPtr pConnection)
{
??? // Print Provider Errors from Connection object.
??? // pErr is a record object in the Connection's Error collection.
??? ErrorPtr? pErr = NULL;
??? if( (pConnection->Errors->Count) > 0)
??? {
??????? long nCount = pConnection->Errors->Count;
??????? // Collection ranges from 0 to nCount -1.
??????? for(long i = 0; i < nCount; i++)
??????? {
??????????? pErr = pConnection->Errors->GetItem(i);
??????????? printf("/t Error number: %x/t%s", pErr->Number,
??????????????? pErr->Description);
??????? }
??? }
}
//
//????? PrintComError Function????? //
//
void PrintComError(_com_error &e)
{
??? _bstr_t bstrSource(e.Source());
??? _bstr_t bstrDescription(e.Description());
???
??? // Print Com errors.
??? printf("Error/n");
??? printf("/tCode = %08lx/n", e.Error());
??? printf("/tCode meaning = %s/n", e.ErrorMessage());
??? printf("/tSource = %s/n", (LPCSTR) bstrSource);
??? printf("/tDescription = %s/n", (LPCSTR) bstrDescription);
}
// EndExecuteCpp
?
發(fā)信人: wesley (pine~心中的森林), 信區(qū): VisualC
標(biāo)? 題: ADO is AOK (ADO其實(shí)很簡單) (一) (轉(zhuǎn)載)
發(fā)信站: BBS 水木清華站 (Fri Nov 19 11:30:18 1999)
?
【 以下文字轉(zhuǎn)載自 Database 討論區(qū) 】
【 原文由 wesley 所發(fā)表 】
本文摘自http://www.codeguru.com/mfc_database/Ado_Aok.shtml。作者:Bob Place
?
翻譯改寫:wesley at 水木清華。 Email: wesley@video.mdc.tsinghua.edu.cn。
?
前言:
??? 有人經(jīng)常問:現(xiàn)在最好的數(shù)據(jù)訪問方法是什么?回答當(dāng)然是ADO!
??? M$花了不少時間,推出了一種叫 UDA(Universal Data Access)的東東,還有一套看
起來蠻簡單的數(shù)據(jù)訪問對象ADO(ActiveX Data Object),用來替代過時的 DAO(Data Ac
cess Object)、RDO(Remote Data Object)。
??? DAO 的底層是 JET 引擎,主要用來提供對 ACCESS 數(shù)據(jù)庫的訪問,比較新的版本也
支持訪問其他數(shù)據(jù)庫,不過對于其他數(shù)據(jù)庫,需經(jīng)過 JET 的中間層,訪問速度比較差。
在所有對 ACCESS 數(shù)據(jù)庫的訪問方法中, JET是最快的。最新的 JET Engine版本為4.0
,對應(yīng)的 DAO 版本為3.6,可以訪問 ACCESS 2000 的數(shù)據(jù)庫。MFC里面 CDAODatabase
和 CDAORecordset 即為 DAO 的 MFC 包裝。
??? RDO 的底層是 ODBC,RDO僅僅是對ODBC API的一個薄包裝層,薄得簡直有點(diǎn)不象樣
,很多人都用 ODBC API寫過程序 (我的第一個SQL Server客戶端程序就是用ODBC API寫
的一個Console Application),把那些并不復(fù)雜的API跟RDO一比就能發(fā)現(xiàn),包裝得簡直
沒有專業(yè)精神。也正因?yàn)楸?#xff0c;所以速度較快,在ADO出現(xiàn)以前,訪問MS SQL Server最快
的方法就是 RDO 了(不要跟我說還有DB-Library,不論是看MS的文檔,還是我親自實(shí)驗(yàn)
,DB-Library都沒有ODBC/RDO快)。最新的 RDO 版本為 2.0,說是最新,似乎也有好幾
年沒有更新了,原因是MS早已經(jīng)決定將其淘汰。不過遺憾的是,現(xiàn)在大家都還在用的 C
Database、CRecordset 就是 RDO 的 MFC 包裝。現(xiàn)在還用這兩個類,就象有了寶馬奔馳
,還騎破永久上下班,只是因?yàn)椴粫_車。 :-(
??? ADO 的底層是 OLE DB,不僅能訪問關(guān)系型數(shù)據(jù)庫,也可以訪問非關(guān)系型數(shù)據(jù)庫,這
可是現(xiàn)在最快速的數(shù)據(jù)庫訪問中間層啊!ADO對OLE DB的包裝可以說相當(dāng)成功,對象模型
簡明扼要,沒有一點(diǎn)多余的東西,功能還遠(yuǎn)超DAO、RDO。直到此時,我才算是有點(diǎn)佩服
MS,OLE DB里面數(shù)十個密密麻麻的接口對我來說實(shí)在是太恐怖了,還是乖乖的用ADO吧。
?
??? 有一點(diǎn)很不幸,微軟提供的ADO文檔幾乎沒有有關(guān)VC的內(nèi)容,象我這樣的VC菜鳥,坐
進(jìn)了寶馬舒適的駕駛倉,不知道該怎么上手下腳,郁悶之極!
??? 這篇文章真是救黎民于水火之中,不容我不把它貢獻(xiàn)出來讓大家共享。
?
開始:
??? 在用ADO以前,一定得讓你的程序知道去哪里找ADO。在stdafx.h文件里,需要加上
下面的代碼:
?#import "c:/program files/common files/system/ado/msado15.dll" no_namespace
?rename("EOF", "adoEOF")
??? 這行代碼的作用是,告訴編譯器去哪里找ADO的庫文件(可能在你的機(jī)器上路徑有所
不同),然后說明不用namespace,并且將 EOF 更名為 adoEOF(如果不這樣干,很有可能
會碰到常量沖突)。
??? 只要加了這句話,準(zhǔn)備工作就全干完了,很簡單是嗎?不用包含任何頭文件,不用
為link指定任何lib文件,我也覺得有點(diǎn)神奇。//shrug
?
_ConnectionPtr, _CommandPtr, 和 _RecordsetPtr (本文中未提及 _CommandPtr):
??? ADO,和 CDAODatabase、CDatabase 非常相似,也分這么幾塊,不同的是,ADO 以
?COM 為基礎(chǔ),這幾塊都是標(biāo)準(zhǔn)的COM組件,而 CDatabase 等等則是 MFC 類。有一點(diǎn)必
須提請注意,要學(xué)習(xí)ADO編程,學(xué)點(diǎn)COM是不可避免的了,不過這是件好事,現(xiàn)在如果不
會一點(diǎn)COM、OLE什么的,估計(jì)很難適應(yīng)Windows編程的形勢。ADO里面的三個組成部份就
是三個COM組件:Connection、Command、Recordset。(還有兩個暫時用不上的:)
??? Connection用于建立數(shù)據(jù)庫連接,執(zhí)行不返回任何結(jié)果集的SQL語句。
??? Command用于返回結(jié)果集,并提供簡單的方法執(zhí)行存儲過程或者任何返回結(jié)果集的S
QL語句。
??? Recordset就是結(jié)果集,可進(jìn)行數(shù)據(jù)的存取、滾動操作。
??? 如果給Command和Recordset正確的Connection string,而不是一個Connection對象
的指針,它們一樣可以打開記錄集,這種情況適用于單數(shù)據(jù)庫操作。當(dāng)程序里需要頻繁
進(jìn)行數(shù)據(jù)庫操作時,最好還是預(yù)先定義一個Connection對象,用它連接數(shù)據(jù)庫,而用Re
cordset處理數(shù)據(jù)。本文中將大量討論這兩個對象。
??? _ConnectionPtr是一個Connection的接口,與CDatabase和CDAODatabase類似,實(shí)際
工作原理也差不多。在程序里創(chuàng)建它的實(shí)例,通過某個OLE DB provider指向一個數(shù)據(jù)源
,并開啟連接。下面的代碼是CDAODatabase和_ConnectionPtr的開啟實(shí)例:
?
DAO:
??? CDaoDatabase MyDb = new CDaoDatabase();
??? m_DaoServerDB.Open(NULL,FALSE,FALSE,"ODBC;DSN=SAMS_SVR;UID=admin;PWD=adm
in");
?
ADO:
??? _ConnectionPtr MyDb;
??? MyDb.CreateInstance(__uuidof(Connection));
??? MyDb->Open("DSN=SAMS_SVR;UID=admin;PWD=admin","","",-1);
?
??? _RecordsetPtr是記錄集接口,與CDAORecordset類似。先看看它的開啟方式與CDAO
Recordset有多么相似:
?
DAO:
??? CDaoRecordset MySet = new CDaoRecordset(MyDb);
??? MySet->Open(AFX_DAO_USE_DEFAULT_TYPE,"SELECT * FROM some_table");
?
ADO:
??? _RecordsetPtr MySet;
??? MySet.CreateInstance(__uuidof(Recordset));
??? MySet->Open("SELECT * FROM some_table", MyDb.GetInterfacePtr(), adOpenDy
namic, adLockOptimistic, adCmdText);
?
??? (譯者注:請注意ADO在Open Recordset的時候,使用了MyDb.GetInterfacePtr()作
為參數(shù)之一,當(dāng)時我用了_ConnectionPtr, &(_ConnectionPtr)都不行,原來要這么用,
真是不服不行。:()
??? ADO只是略微的麻煩一點(diǎn),不過花這點(diǎn)功夫獲得ADO的多多好處還是很值的。
??? 現(xiàn)在有了一個Connection和一個Recordset,下面該用這兩個東東取點(diǎn)數(shù)據(jù)出來了。
不妨先假定有一個名為m_List的Listbox,我們把數(shù)據(jù)取出來往里塞。
?
DAO:
??? VARIANT *vFieldValue;
??? COleVariant covFieldValue;
??? CString Holder;
??? while(!MySet->IsEOF())
??? {
??????? MySet->GetFieldValue("FIELD_1", covFieldValue);
??????? vFieldValue = (LPVARIANT)covFieldValue;
??????? if(vFieldValue->vt!-VT_NULL)
??????? {
??????????? Holder.Format("%s",vFieldValue->pbVal);
??????????? m_List.AddString(Holder);
??????? }
??????? MySet.MoveNext();
??? }
?
ADO:
??? _variant_t Holder
??? while(!MySet->adoEOF)
??? {
??????? Holder = MySet->GetCollect("FIELD_1");
??????? if(Holder.vt!=VT_NULL)
??????? m_List.AddString((char*)_bstr_t(Holder));
??????? MySet->MoveNext();
??? }
?
??? 注意:在微軟所有的文檔里,都沒找到GetCollect方法。我找了所有的地方,也從
沒人提起過,在文檔里的示例方式是這樣:
??????? Holder = MySet->GetFields->Field->(_variant_t(FieldNumber))->Value;
??? 你喜歡哪一種呢,反正我是喜歡GetCollect。(譯者注:鬼知道他從哪里搞到這個方
法,難道是從MS自己人的程序里?I 服了 him。)
?
動態(tài)綁定 vs DFX(CRecordset 和 CDAORecordset的預(yù)先字段綁定,Data Field Exchan
ge):
??? 動態(tài)綁定即使用SQL語句動態(tài)構(gòu)造結(jié)果字段,而不是象 CDAORecordset 里面用DFX去
把所有原始字段映射成成員變量。動態(tài)綁定的例子:
??? SELECT (SUM(field_1) + SUM(field_2)) AS answer FROM some_table
??? 如果用DFX,估計(jì)就得在程序里自己寫了:
??? m_answer = m_field_1 + m_field2;
??? (譯者注:盡管很多人喜歡 DFX 的字段預(yù)先綁定,寧可這么寫代碼,不過我很少用
?DFX,當(dāng)然在 VB 里面更是從來不會用到,好象用得比較多的也就是在 Delphi 里面,
用起來還比較爽,大概是 MS 的文檔里讓我別用,而 Borland 的破文檔里卻沒說吧。
:-)
??? 相比之下,動態(tài)綁定確有其優(yōu)越的地方,減少了代碼量,也讓程序更小、更易維護(hù)
。再說,這也是MS推薦的獲取數(shù)據(jù)方式,更靈活,速度更快,更易維護(hù),我們還能要求
什么呢?
??? 對于大多數(shù)程序來說,都是創(chuàng)建一個全局的Connection,然后用Recordset來處理數(shù)
據(jù),如果Recordset數(shù)量很多,可以想象光是花費(fèi)在DFX上面的代碼就有多少。如果使用
動態(tài)綁定,這些都省掉了。
?
_variant_t 和 _bstr_t 到底是什么玩意?
??? 很不幸,我們喜愛的CString類在COM里用不了(CStringEx也一樣),因?yàn)镃OM必須設(shè)
計(jì)成跨平臺,它需要一種更普遍的方式來處理字符串以及其他數(shù)據(jù)。這就是VARIANT數(shù)據(jù)
類型的來歷,還有BSTR類型。VARIANT就是一個巨大的 union,包含了你能想得到的所有
的數(shù)據(jù)類型,除了char*,不過還好,BSTR取代了char*。
??? (譯者注:似乎VARIANT是個很慢的東西,大家都不愿意使它,不過按我看來,情況
沒這么糟糕,union照理說不應(yīng)該慢到哪去,要說慢,也是慢在給VARIANT分配地址空間
上,這點(diǎn)在VC里面做得比VB要好
??? 這些東西看起來的確有點(diǎn)恐怖,不過實(shí)在用不著怕,等下面熟悉了這兩個東西之后
,你會很快喜歡的)
??? 簡單來說,_variant_t是一個類,包裝了VARIANT數(shù)據(jù)類型,并允許我們簡單的對之
進(jìn)行強(qiáng)制類型轉(zhuǎn)換(相信大家都喜歡這個),_bstr_t對BSTR干了同樣的事情。在下面的例
子里,你將看到怎么用GetCollect把數(shù)據(jù)取到VARIANT里,又怎么把它放到_bstr_t里,
最后強(qiáng)制轉(zhuǎn)換成char*,以及把_variant_t強(qiáng)制轉(zhuǎn)換成long、double或者其他一切東西:
?
??? _variant_t Holder;
??? // first get the VARIANT and put it into the _variant_t
??? Holder = MySet->GetCollect("FIELD_1");
??? // now put it into a _bstr_t and cast it to a char*
??? m_List.AddString((char*)_bstr_t(Holder));
?
??? 對比一下沒有用 _variant_t 和 _bstr_t 的代碼:
?
??? COleVariant covFieldValuel
??? VARIANT vFieldValue
??? CString Holder;
??? MySet->GetFieldValue("FIELD_1", covFieldValue);
??? vFieldValue = (LPVARIANT)covFieldValue;
??? Holder.Format("%s",vFieldValue->pbVal);
??? m_List.AddString(Holder);
?
??? 區(qū)別大了!
?
(待續(xù))
?
--
世界上既無所謂幸福也無所謂不幸
只有一種狀況和另一種狀況的比較
只有體驗(yàn)過極度不幸的人
才能品嘗到極度的幸福
?
※ 來源:·BBS 水木清華站 bbs.net.tsinghua.edu.cn·[FROM: 166.111.159.21]
發(fā)信人: wesley (pine~心中的森林), 信區(qū): VisualC
標(biāo)? 題: ADO is AOK (ADO其實(shí)很簡單) (二) (轉(zhuǎn)載)
發(fā)信站: BBS 水木清華站 (Fri Nov 19 11:30:29 1999)
?
【 以下文字轉(zhuǎn)載自 Database 討論區(qū) 】
【 原文由 wesley 所發(fā)表 】
Update,Insert,Delete:
??? 當(dāng)我進(jìn)行Update,Insert,Delete操作時,通常我喜歡用Connection和Command對象
,原因是,用CString來構(gòu)造SQL語句簡單一些,然后直接用Connection.Execute就行了
。當(dāng)然,用Recordset干這些也是可以的。
?
??? Update方法有下面三種方法調(diào)用:
??? 1: 給某個Field對象(或某些個)的Value屬性賦值,然后調(diào)用Update方法;
??? 2: 將字段名和字段值作為參數(shù)傳給Update方法;
??? 3: 將字段名和字段值的數(shù)組作為參數(shù)傳給Update方法。
?
??? AddNew方法如下調(diào)用:
??? 1: 直接調(diào)用,然后同Update調(diào)用方法一;
??? 2: 將字段名數(shù)組、字段值數(shù)組作為參數(shù)傳給AddNew方法。
?
??? Delete方法最簡單,直接調(diào)用就行了,刪除當(dāng)前記錄!
?
??? 做完這些事情,可能需要調(diào)用Requery方法才能看到效果。
??? 下面的示例代碼需要我們創(chuàng)建一個簡單的MFC Application,然后在CWinApp類里面
聲明三個接口的對象:
?
??? // Global ADO Objects
??? // connection
??? _ConnectionPtr m_pConnection;
??? _CommandPtr? m_pCommand;
??? _RecordsetPtr m_pRecordset;
?
??? 在VC6里面有一點(diǎn)很有意思,如果敲 "m_pConnection.",你會看到一個方法和屬性
列表,如果敲"m_pConnection->",還會看到一個方法和屬性列表,當(dāng)然里面的內(nèi)容完全
不同,因?yàn)槟銓?shí)際是在指向兩個不同的東西。下面就是這兩種混用的代碼:
?
??? _ConnectionPtr MyDb;
??? MyDb.CreateInstance(__uuidof(Connection));
??? MyDb->Open("DSN=SAMS_SVR;UID=admin;PWD=admin","","",-1);
?
??? 回到示例代碼,在 application 的 InitInstance 方法里,我打開數(shù)據(jù)連接,指向我
機(jī)器上的一個數(shù)據(jù)庫,你需要更改ConnectionString,使用你自己的ODBC數(shù)據(jù)源或指定
一個OLE DB provider。
?
??? // When we open the application we will open the ADO connection
??? m_pConnection.CreateInstance(__uuidof(Connection));
??? m_pConnection->Open("DSN=ADOTest","","",-1);
?
??? 如果你打開about對話框,就會看到一個Listbox,還有一個叫button1的按鈕,這里
面包含了 ADO 調(diào)用的核心代碼。我創(chuàng)建了一個_RecordsetPtr接口的實(shí)例,打開我需要
的記錄集,然后遍歷所有記錄,將它們?nèi)絃istbox里去:
?
??? _variant_t TheValue;
??? theApp.m_pRecordset.CreateInstance(__uuidof(Recordset));
??? try
??? {
??????? theApp.m_pRecordset->Open("SELECT DISTINCT FLDESC FROM tblFALines",
??????????????? theApp.m_pConnection.GetInterfacePtr(),
??????????????? adOpenDynamic,
??????????????? adLockOptimistic,
??????????????? adCmdText);
??????? while(!theApp.m_pRecordset->adoEOF)
??????? {
??????????????? TheValue = theApp.m_pRecordset->GetCollect("FLDESC");
??????????????? if(TheValue.vt!=VT_NULL)
??????????????? m_List.AddString((char*)_bstr_t(TheValue));
??????????????? theApp.m_pRecordset->MoveNext();
??????? }
??????? theApp.m_pRecordset->Close();
??? }
??? catch(_com_error *e)
??? {
??????? CString Error = e->ErrorMessage();
??????? AfxMessageBox(e->ErrorMessage());
??? }
??? catch(...)
??? {
??????? MessageBox("Whoa this is bad");
??? }
?
??? 記得一定要用try和catch,否則ADO調(diào)用錯誤有可能使你的程序崩潰,一定要隨時記
得捕捉_com_error例外以及其它錯誤。
??? 我盡可能的使代碼簡單,所以省略了很多細(xì)節(jié),尤其是忽略了很多好的編程習(xí)慣(比
如檢查大多數(shù)COM方法都返回的HRESULT值)。本文的目的是想說,ADO并沒什么難的,CO
M也一樣,而不是想表現(xiàn)ADO能做的所有事情。我甚至都沒仔細(xì)想過ADO能為你帶來什么,
不過我肯定一點(diǎn),那就是ADO比DAO、RDO更快、更容易使用、并且功能強(qiáng)大得多。看看本
站點(diǎn)其它的文章,你就會知道,通過ADO調(diào)用存儲過程有多么容易!
??? 最后,我想向大家推薦兩本書,其中有一本是完全免費(fèi)的,在www.informit.com可
以找到電子版,另一本必須得付錢買了,不過我還是推薦兩本都買,除非你家浴池里、
床頭邊也放了計(jì)算機(jī)。 :)
??? 免費(fèi)的那本是<Learn Database Programming with Visual C++ in 21 days>。哦,
我知道你在想什么,我也知道那些'in 21 days'的書通常都很爛,而且當(dāng)著其它程序員
的面買這樣的書的確有點(diǎn)丟面子,并且還很不好意思把這樣的書擺在書架上顯眼的位置
。不過這一本絕對是個例外!里面的內(nèi)容簡直太棒了!
??? 要花錢的那本是<ADO 2.0>,由WROC出版... (譯者注:細(xì)節(jié)就省了吧,反正我們也
不會花美元去買英文書)。
?
??? 譯者注:一年以前,我在www.mcp.com(現(xiàn)在已經(jīng)更名為informit)看到這本書,看過
之后覺得實(shí)在是本難得一見的好書,不僅講ADO深入淺出,COM的來歷、基礎(chǔ)、發(fā)展都說
得很清楚,還有不少C++的高級知識,這也是我看過的唯一一本“21天”的書,與其它的
“21天”簡直是天壤之別。強(qiáng)烈建議大家買一本。
??? 中譯本:Visual C++ 數(shù)據(jù)庫編程...(一時記不起了),清華大學(xué)出版社出版,清華
書店有賣,大家快去買啊!注意不要買機(jī)械工業(yè)出版社那本 VC 數(shù)據(jù)庫編程的書,看起
來名字很象,不過內(nèi)容爛得可以。
??? 如果要看相關(guān)的文章,請?jiān)L問站點(diǎn)www.codeguru.com,祝大家早日成為編程高手,
已經(jīng)是高手的更加高!
?
--
世界上既無所謂幸福也無所謂不幸
只有一種狀況和另一種狀況的比較
只有體驗(yàn)過極度不幸的人
才能品嘗到極度的幸福
?
※ 修改:·wesley 於 Nov 19 11:30:06 修改本文·[FROM:? 166.111.159.21]
※ 來源:·BBS 水木清華站 bbs.net.tsinghua.edu.cn·[FROM: 166.111.159.21]
在stdafx.h中添加:
#import "c:/program files/common files/system/ado/msado15.dll" no_namespace rename("EOF","adoEOF")
正式代碼:
_RecordsetPtr m_pRs;
_ConnectionPtr m_pConn;
try
{
?CoInitialize(NULL);
?m_pConn.CreateInstance(__uuidof(Connection));
?m_pRs.CreateInstance(__uuidof(Recordset));
?
?//設(shè)置服務(wù)器端游標(biāo)
?m_pConn->CursorLocation = adUseServer;//adUseClient;adUseNone
???
?//連接ORACLE數(shù)據(jù)庫
?m_pConn->Open(L"Provider=OraOLEDB.Oracle.1;User ID=pkuwh_za;password=a;Data Source=213;Persist Security Info=False",
??????L"",L"",adOpenUnspecified);
?//上面的語句按下面的方式寫可以不用配置oracle數(shù)據(jù)源
??????? //m_pConn->Open(L"Provider=OraOLEDB.Oracle.1;User ID=pkuwh_za;Password=a;Data Source=(DESCRIPTION=(ADDRESS=(PROTOCOL=TCP)(HOST=192.168.3.213)(PORT=1521))(CONNECT_DATA=(SERVICE_NAME = ZAGL)));Persist Security Info=False",L"",L"",adOpenUnspecified);
?
?m_pRs->PutRefActiveConnection(m_pConn);
?CString t = "select * from case_acceptinfo where rownum <= 3";
?m_pRs->Open(_variant_t(t),_variant_t((IDispatch*)m_pConn,true),adOpenDynamic,adLockOptimistic,adCmdUnknown);
?
?while(!m_pRs->adoEOF)
?{
??CString m_s1 = _com_util::ConvertBSTRToString((_bstr_t)m_pRs->GetCollect("reporttime"));
??MessageBox(m_s1);
??m_pRs->MoveNext();
?}
?m_pRs->Close();
?m_pConn->Close();
?m_pRs=NULL;
?m_pConn=NULL;
?CoUninitialize();
}
catch(_com_error &e)
{?
?AfxMessageBox(e.Description());
}
總結(jié)
- 上一篇: Android手机应用开发(一) | 基
- 下一篇: jquery选择器篇