- 在线时间
- 0 小时
- 最后登录
- 2007-11-12
- 注册时间
- 2004-12-24
- 听众数
- 2
- 收听数
- 0
- 能力
- 0 分
- 体力
- 2467 点
- 威望
- 0 点
- 阅读权限
- 50
- 积分
- 882
- 相册
- 0
- 日志
- 0
- 记录
- 0
- 帖子
- 205
- 主题
- 206
- 精华
- 2
- 分享
- 0
- 好友
- 0
升级   70.5% 该用户从未签到
 |
1. 生成应用程序框架并初始化OLE/COM库环境 ! c4 [+ O# I, z( d0 s2 m+ C
5 E) d* f6 N. g
创建一个标准的MFC AppWizard(exe)应用程序,然后在使用ADO数据库的InitInstance函数中初始化OLE/COM库(因为ADO库是一个COM DLL库)。
+ O8 b. b( ~; ]6 b4 \) e本例为:
" S" U/ n E0 C' J) {) `2 C+ G3 ^/ o9 h# P; Z; k
BOOL CAdotestDlg::OnInitDialog()7 i+ M) h' I3 N7 |$ K* i& i+ ~
{" [) T2 X" S# ^3 I. @! P% F/ u* v$ p& @
::CoInitialize(NULL); //初始化OLE/COM库环境 ' q& M4 r( o2 B
} : d# I2 ?$ t6 ^( l0 w7 x
% _; O) L2 c* ?! f k: |' S
程序最后要调用 ::CoUninitialize();//释放程序占用的COM 资源。
5 Q$ O/ l* B3 R! d6 X8 V) ~( m1 P
另外:) j7 ^: e9 y5 @: @
* R, P5 U6 O+ w
m_pRecordset->Close(); 注意!!!不要多次关闭!!!!!!!!!!!!
, z+ ?8 p( J" }1 f. Cm_pConnection->Close();
. g2 `) C) V1 p' a. m& L: sm_pRecordset = NULL;- ]4 ~5 b) E- }- m
m_pConnection = NULL;
+ i$ J/ _7 ^9 r5 t4 J1 q5 e 2. 引入ADO库文件 6 `/ U( i- z* _; Y& g$ M+ ^
) P+ e9 Z3 V0 m5 n( h 使用ADO前必须在工程的stdafx.h文件最后用直接引入符号#import引入ADO库文件,以使编译器能正确编译。代码如下:& B4 R' U6 y/ h' i0 l
#import "C:\Program Files\common files\system\ado\msado15.dll" no_namespace rename("EOF","adoEOF")
( W$ b0 N- c' j* q% Q& f- q, c6 D ADO类的定义是作为一种资源存储在ADO DLL(msado15.dll)中,在其内部称为类型库。类型库描述了自治接口,以及C++使用的COM vtable接口。当使用#import指令时,在运行时Visual C++需要从ADO DLL中读取这个类型库,并以此创建一组C++头文件。这些头文件具有.tli 和.tlh扩展名,读者可以在项目的目录下找到这两个文件。在C++程序代码中调用的ADO类要在这些文件中定义。
8 o8 A7 I+ K, O% q, e' u. r9 d! | 程序的第三行指示ADO对象不使用名称空间。在有些应用程序中,由于应用程序中的对象与ADO中的对象之间可能会出现命名冲突,所以有必要使用名称空间。如果要使用名称空间,则可把第三行程序修改为: rename_namespace("AdoNS")。第四行代码将ADO中的EOF(文件结束)更名为adoEOF,以避免与定义了自己的EOF的其他库冲突。
0 k2 W6 J7 ]4 d- w: W2 Y: O# O+ |" H
) U0 Y- `! e, W- L+ R 3.利用智能指针进行数据库操作 2 b* t, |; i- P' b
) }( P' G; H/ B& b6 T" u9 S. W* `2 | 在CaboutDlg头文件中定义两个ADO智能指针类实例,并在对话框中加入一个ListCtrl。
3 @7 {4 d. C8 l R& I+ h
$ ~/ g# x3 j3 S+ q8 Tclass CAdotestDlg : public CDialog. E( \# L" ]$ P* q: p" \# A- w
{7 }% {$ Y9 t; m+ A: Z+ [
_ConnectionPtr m_pConnection;% k# w `# b$ E# a" @6 F
_RecordsetPtr m_pRecordset;
" c1 {$ K* U6 x* p: H9 } ClistCtrl m_List; ( i: Z) I4 d" K5 |( c9 D3 B# H# P
......3 [( s. l/ v j# |* V$ S% H! p$ w1 q1 Y
} / A! M3 ~; _0 m+ V3 b& T
0 [& G; C ~) h+ v' n0 B/ T+ p. E ADO库包含三个智能指针:_ConnectionPtr、_CommandPtr和_RecordsetPtr。
. d6 _7 J3 Y& B9 a7 s2 B( t( D
! \% z K9 g$ z* J9 c _ConnectionPtr通常被用来创建一个数据连接或执行一条不返回任何结果的SQL语句,如一个存储过程。
1 p! r% d7 E* G/ u _CommandPtr返回一个记录集。它提供了一种简单的方法来执行返回记录集的存储过程和SQL语句。在使用_CommandPtr接口时,可以利用全局_ConnectionPtr接口,也可以在_CommandPtr接口里直接使用连接串。 _RecordsetPtr是一个记录集对象。与以上两种对象相比,它对记录集提供了更多的控制功能,如记录锁定、游标控制等。
; r- o) R" i. v( {
' D# r% X+ r: ] 在使用ADO程序的事件响应中OnButton1加入以下代码: 3 v( f. Z! X0 Z6 N" z
8 S# g8 g- L2 K- i8 l' d
void CAdotestDlg::OnButton1() + f6 X" i( g& g; H7 z' X9 S
{
: W* \& S: Z8 M, v& S) gm_List.ResetContent();
- Z# t, d& M! F% am_pConnection.CreateInstance(_uuidof(Connection)); //初始化Connection指针
0 R: V: Q5 {- @2 E4 q4 Um_pRecordset.CreateInstance(_uuidof(Recordset));//初始化Recordset指针! ?3 |. W3 G$ T+ D, `+ |1 i* D* }* M
try& B/ S/ P" k% g" j1 F
{
/ V5 P! M$ ?" C! E m_pConnection->Open("DSN=ADOTest","","",0); //连接叫作ADOTest的ODBC数据源9 p. Y/ A- v& B2 f
//注意:这是连接不需要用户ID或密码的open 函数
) D! H+ m' i7 N, a% M3 ~) G // 否则形式为 ->Open("DSN=test;uid=sa;pwd=123;","","",0);
3 C& [: F7 m# |- k% y+ j
# S: M/ ^1 V& G1 R z- d // 执行SQL语句得到一个记录集把其指针赋值给m_pRecordset
6 ]' b5 S, M# }* e CString strSql="select * from middle";
5 x/ p+ b2 C& C" |* ~ BSTR bstrSQL = strSql.AllocSysString(); # T5 J( j6 v! z
m_pRecordset->Open(bstrSQL,(IDispatch*)m_pConnection,adOpenDynamic,adLockOptimistic,adCmdText);
3 V! l( H1 d, [. J+ s& s- ] //adOpenDynamic:动态 adLockOptimistic乐观封锁法 adCmdText:文本查询语句
" _ h$ E! I1 n0 F. d. K8 v2 q while(!m_pRecordset->adoEOF)//遍历所有记录1 T) L3 B& x+ h" s: O1 v$ X7 C
{
# a; r! H4 A) q* p$ b& U //取纪录字段值方式之一
1 s& R6 V7 F. J2 f) Z( m _variant_t Thevalue; //VARIANT数据类型; {1 J! p' g, }
Thevalue = m_pRecordset->GetCollect("BIG_NAME");//得到字段BIG_NAME的值5 B* Z! p; F/ x. K9 w g1 K
if(Thevalue.vt!=VT_NULL)
|$ R# o- p9 J; `. i m_List.AddString((char*)_bstr_t(Thevalue));
2 g; x6 R- ]/ n //将该值加入到列表控件中
! s( t& {' A: U
\# K$ I# a6 j# z$ K. _+ _5 L6 ` //取纪录字段值方式之二( f5 R- K' d- |7 K% b$ L! N
// _bstr_t Thevalue1=m_pRecordset->Fields->GetItem("BIG_NAME")->value;( K1 \) D: X9 c9 a% Q) X i% a1 E
// CString temp=Thevalue1.copy();
4 n" ]7 w y' N+ U; x // m_List.AddString(temp);
* G* m7 J% T/ ^+ a3 Z+ ^. L //数据类型转换
% y( f0 I4 \1 S _variant_t vUsername,vBirthday,vID,vOld;4 q4 J+ q7 d) d$ N1 x n* ?
TRACE("id:%d,姓名:%s,年龄:%d,生日:%s\r\n",
& R4 D5 s: f3 N3 f7 I- a vID.lVal,(LPCTSTR)(_bstr_t)vUsername,vOld.lVal,(LPCTSTR)(_bstr_t)vBirthday);" X$ P; ~, \& J0 u$ z
I3 W# K, S" \/ t7 ~! X m_pRecordset->MoveNext();//转到下一条纪录 D Q9 l0 @- y N$ S D
}' Z, G# W) _, Z0 ]5 p* V
m_pRecordset->Close();5 a* Z: t: c& v, b& B% k
m_pConnection->Close();
p) I0 o, m% W5 B1 G% v( w}* ?1 \8 U; s7 ]+ R! A
catch (_com_error e)//异常处理
" |/ E7 z0 X$ }- m& d! K{
# [; y% E: w: T% r3 K) _2 {/ [* \ AfxMessageBox(e.ErrorMessage());3 D" h5 j" O2 i _, p
}
6 ?/ B3 A# V/ J. v' y. G0 m* D: tm_pRecordset->Close(); //注意!!!不要多次关闭!!!!否则会出错
# c, H; u3 I& a- y! a5 im_pConnection->Close();
: E9 p& U# y( r! g8 |% Sm_pRecordset = NULL;
) g& _8 C' r* W/ Fm_pConnection = NULL;
) U7 E+ H5 U% L9 z( Z/ K0 E}
1 ^: G9 j/ Y# F4 ~( C- _( L1 L
1 _# N4 O7 ?5 } 程序中通过_variant_t和_bstr_t转换COM对象和C++类型的数据, _variant_t类封装了OLE自治VARIANT数据类型。在C++中使用_variant_t类要比直接使用VARIANT数据类型容易得多。
; T/ x' Z: [8 c7 Q* A& N
3 d0 s% Q, ]/ M* Y& Y O 好,编译后该程序就能运行了,但记住运行前要创建一个叫ADOTest的ODBC数据源。该程序将把表middle中的BIG_NAME字段值显示在列表控件中。 |
zan
|