?ODBC(Open Database Connectivity,开放式数据库连接),是一种用来在相关或不相关的数据库管理系统(DBMS)中存取数据的标准应用程序接口(API)。本文给出Windows 95 环境下用Visual C++ 进行ODBC 编程的具体方法及技巧。 1sJN^BvuG
F~AS(sk
---- 关键字:ODBC,Visual C++,Windows 编程。 7y\g~?5N
a*hThr+$M
---- 一.概述 X
A|`wAGP
z,)sS<t(
---- ODBC 是一种使用SQL 的程序设计接口。使用ODBC 让应用程序的编写者避免了与数据源相联的复杂性。这项技术目前已经得到了大多数DBMS 厂商们的广泛支持。 &^H
"T6
aiHr2x6
---- Microsoft Developer Studio 为大多数标准的数据库格式提供了32 位ODBC 驱动器。这些标准数据格式包括有:SQL Server、Access、Paradox、dBase、FoxPro、Excel、Oracle 以及Microsoft Text。如果用户希望使用其他数据格式,用户需要相应的ODBC 驱动器及DBMS。 d/&|%Z
r
\_E.%K
---- 用户使用自己的DBMS 数据库管理功能生成新的数据库模式后,就可以使用ODBC 来登录数据源。对用户的应用程序来说,只要安装有驱动程序,就能注册很多不同的数据库。登录数据库的具体操作参见有关ODBC 的联机帮助。 w-$w
k
))*z FV
---- 二.MFC 提供的ODBC 数据库类 pYG,5+g
*
2%e.d3"M
---- Visual C++ 的MFC 基类库定义了几个数据库类。在利用ODBC 编程时,经常要使用到CDatabase( 数据库类),CRecordSet( 记录集类) 和CRecordView( 可视记录集类)。其中: Uz|]}t5V
O m
---- CDatabase 类对象提供了对数据源的连接,通过它你可以对数据源进行操作。 q9!9OcN2
x,dv~QU
---- CRecordSet 类对象提供了从数据源中提取出的记录集。CRecordSet 对象通常用于两种形式:动态行集(dynasets)和快照集(snapshots)。动态行集能保持与其他用户所做的更改保持同步。快照集则是数据的一个静态视图。每一种形式在记录集被打开时都提供一组记录,所不同的是,当你在一个动态行集里滚动到一条记录时,由其他用户或是你应用程序中的其他记录集对该记录所做的更改会相应地显示出来。 q@9i3*q;
nS4S[|w"
---- CRecordView 类对象能以控制的形式显示数据库记录。这个视图是直接连到一个CRecordSet 对象的表视图。 E2IV R]C2^
f zO8by
---- 三.应用ODBC 编程 -#6*T,f0P(
ArYF\7P
---- 应用Visual C++ 的AppWizard 可以自动生成一个ODBC 应用程序框架。方法是:打开File 菜单的New 选项,选取Projects,填入工程名,选择MFC AppWizard (exe),然后按AppWizard 的提示进行操作。当AppWizard 询问是否包含数据库支持时,如果你想读写数据库,那么选定Database view with file support;而 ?阆敕梦适?菘獾男畔⒍?幌牖匦此?龅母谋洌?敲囱《―atabase view without file support 选项就比较合适了。选择了数据库支持之后Database Source 按钮会激活,选中它去调用Data Options 对话框。在Database Options 对话框中会显示已向ODBC 注册的数据库资源,选定你所要操作的数据库,如:Super_ES,单击OK 后会出现Select Database Tables 对话框,其中列举了你所选中的数据库中包含的全部表,选择你希望操作的表后,单击OK。在选定了数据库和数据表之后,你可以按照惯例继续进行AppWizard 操作。 ];;w/$zke
`1@[uWl
---- 特别需要指出的是:在生成的应用程序框架View 类(如:CSuper_ESView)中包含一个指向CSuper_ESSet 对象的指针m_pSet,该指针由AppWizard 建立,目的是在视表单和记录集之间建立联系,使得记录集中的查询结果能够很容易地在视表单上显示出来。有关m_pSet 的详细用法可以参见Visual C++ Online Book。 W<VHv"?V
!&lPdEc@T
---- 程序与数据语言建立联系,使用CDatebase::OpenEx() 或CDatabase::Open() 函数来进行初始化。数据库对象必须在你使用它构造一个记录集对象之前被初始化。 B6\VxSX4{
(Y)h+}n5N
---- 下面举例说明在Visual C++ 环境中ODBC 的编程技巧: ]Qr8 wa>Z
;l ()3;
---- 1 .查询记录 oDUMoX%4s
oNZW#<K
---- 查询记录使用CRecordSet::Open() 和CRecordSet::Requery() 成员函数。在使用CRecordSet 类对象之前,必须使用CRecordSet::Open() 函数来获得有效的记录集。一旦已经使用过CRecordSet::Open() 函数,再次查询时就可以应用CRecordSet::Requery() 函数。在调用CRecordSet::Open() 函数时,如果已经将一个已经打开的CDatabase 对象指针传给CRecordSet 类对象的m_pDatabase 成员变量,则使用该数据库对象建立ODBC 连接;否则如果m_pDatabase 为空指针,就新建一个CDatabase 类对象并使其与缺省的数据源相连,然后进行CRecordSet 类对象的初始化。缺省数据源由GetDefaultConnect() 函数获得。你也可以提供你所需要的SQL 语句,并以它来调用CRecordSet::Open() 函数,例如: \<B6>
!@{[I:5
Super_ESSet.Open(AFX_DATABASE_USE_DEFAULT,strSQL); SZ{cno1`
---- 如果没有指定参数,程序则使用缺省的SQL 语句,即对在GetDefaultSQL() 函数中指定的SQL 语句进行操作: H>f{3S-%
6W;kIoB
CString CSuper_ESSet::GetDefaultSQL() 9 Zm<1Fw
{return _T("[BasicData],[MainSize]");} E{s|#
---- 对于GetDefaultSQL() 函数返回的表名,对应的缺省操作是SELECT 语句,即: l|A8AuO*?
zDyeAxh4
SELECT * FROM BasicData,MainSize x Ui!|c
---- 查询过程中也可以利用CRecordSet 的成员变量m_strFilter 和m_strSort 来执行条件查询和结果排序。m_strFilter 为过滤字符串,存放着SQL 语句中WHERE 后的条件串;m_strSort 为排序字符串,存放着SQL 语句中ORDER BY 后的字符串。如: QJWES%m`
&o@5%Rz2/
Super_ESSet.m_strFilter="TYPE=电动机"; k+$4?/A
Super_ESSet.m_strSort="VOLTAGE"; 8
-;ZPhN&
Super_ESSet.Requery(); D~Ohw sL4
对应的SQL语句为: F |81i$R
SELECT * FROM BasicData,MainSize +c`C9RXk
WHERE TYPE=电动机 v6?\65w,|
ORDER BY VOLTAGE m1i+{((
---- 除了直接赋值给m_strFilter 以外,还可以使用参数化。利用参数化可以更直观,更方便地完成条件查询任务。使用参数化的步骤如下: yQ{_\t1Wd
R"gm]SQ/
---- (1) .声明参变量: P&0cF{
lhl0
CString p1; JK"uj%
float p2; .oj" ru
---- (2) .在构造函数中初始化参变量 ' u};z:t
#.+*G`m
p1=_T(""); ]wpYxos
p2=0.0f; }]+}Tipd
m_nParams=2; >5O y^u6Ly
---- (3) .将参变量与对应列绑定 Z'dI!8(Nf
r/sRXM:3cZ
pFX- >SetFieldType(CFieldExchange::param) Ko|xEz=
RFX_Text(pFX,_T("P1"),p1); E)wT+\
RFX_Single(pFX,_T("P2"),p2); *9Js:z7I
---- 完成以上步骤之后就可以利用参变量进行条件查询了: #4 &N0IG
1r&
?J.z25
m_pSet- >m_strFilter="TYPE=? AND VOLTAGE=?"; |/=p
m_pSet- >p1=" 电动机"; n UCk0:{
m_pSet- >p2=60.0; 9c}]:3#XO
m_pSet- >Requery(); ?>jArzI
---- 参变量的值按绑定的顺序替换查询字串中的"?" 适配符。 5zw23!
)|R0_9CLV
---- 如果查询的结果是多条记录的话,可以用CRecordSet 类的函数Move(),MoveNext(),MovePrev(),MoveFirst() 和MoveLast() 来移动光标。 1vK(^u[
[pgkY!R?)
---- 2 .增加记录 OXX(OCG>
w^E]N
---- 增加记录使用AddNew() 函数,要求数据库必须是以允许增加的方式打开: GdeR#%z
R
4QwWSBJ
m_pSet- >AddNew(); //在表的末尾增加新记录
e=)*O
m_pSet- >SetFieldNull(&(m_pSet- >m_type), FALSE); W#7-%oT
m_pSet- >m_type=" 电动机"; ;:\,x
... //输入新的字段值 /\uH[[s
m_pSet- > Update(); //将新记录存入数据库 -_*XhD
m_pSet- >Requery(); //重建记录集 B
m@oB2x)
---- 3 .删除记录 TgE.=` "7
f9XO9N,hE:
---- 直接使用Delete() 函数,并且在调用Delete() 函数之后不需调用Update() 函数: :G=1$gb
4i\aW:_'i
m_pSet- >Delete(); ^=Tu>{uD
if (!m_pSet- >IsEOF()) h8= MVh(I
m_pSet- >MoveNext(); <T.#A8c
else _B7?C:8Q-
m_pSet- >MoveLast(); YSz$` 7i
---- 4 .修改记录 pkV\D
0R 5^p
---- 修改记录使用Edit() 函数: 2td|8vDA
-kri3?Y,
m_pSet- >Edit(); //修改当前记录 l)PFzIz=V
m_pSet- >m_type="发电机"; vua1iN1
//修改当前记录字段值 CE7pg&dJ)i
... e9hVX[uq
m_pSet- >Update(); //将修改结果存入数据库 `MYK XBM
m_pSet- >Requery(); `Y({#U
---- 5 .撤消操作 HD8"=7zJk
grfdvN
---- 如果用户选择了增加或者修改记录后希望放弃当前操作,可以在调用Update() 函数之前调用: VDu
.L8
aU]O$Pg{
CRecordSet::Move(AFX_MOVE_REFRESH); Z=Y_;dS9
---- 来撤消增加或修改模式,并恢复在增加或修改模式之前的当前记录。其中的参数AFX_MOVE_REFRESH 的值为零。 q,,>:]f#
\%?8jQ'tX
---- 6 .数据库连接的复用 t"bPKFRy9E
ocA'goI-
---- 在CRecordSet 类中定义了一个成员变量m_pDatabase: I1 R\Ts@
nK%/tdq
CDatabase* m_pDatabase; n.Eoi4jV'
---- 它是指向对象数据库类的指针。如果在CRecordSet 类对象调用Open() 函数之前,将一个已经打开的CDatabase 类对象指针传给m_pDatabase,就能共享相同的CDatabase 类对象。如: {L-aXe{
a(43]d&
CDatabase m_db; Gp3nR<+
CRecordSet m_set1,m_set2; `ToRkk&&>{
m_db.Open(_T("Super_ES"));//建立ODBC连接 o`T<