友情提示:如果本网页打开太慢或显示不完整,请尝试鼠标右键“刷新”本网页!
VC语言6.0程序设计从入门到精通-第53部分
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部! 如果本书没有阅读完,想下次继续接着阅读,可使用上方 "收藏到我的浏览器" 功能 和 "加入书签" 功能!
…………………………………………………………Page 365……………………………………………………………
Visual C++ 6。0 程序设计从入门到精通
//根据连接字符串开启数据连接,得到结果集
m_recordSet…》Open(〃select * from Contact〃; m_connection。GetInterfacePtr(); adOpenDynamic;
adLockOptimistic; adCmdText);
注意:当比较频繁的访问数据库的时候,最好先用全局_ConnectionPtr 接口创建一个数据连接,
然后用_ResultsetPtr 接口来处理数据。本章第 6 节的例子便是采用的这种方法。
获得记录集后,可以利用 MoveNext() 函数方便的实现记录集的遍历,代码如下:
while(!m_Recordset…》adoEOF)
{
…
//获得下一条记录
m_Recordset…》MoveNext();
}
当然,利用 12。5 节介绍的其他方法也可以较为方便地实现记录集的遍历,读者可以自己
编码实验。
12。6。4 对记录的操作
对记录的操作主要有添加、修改和删除 3 种。
1.添加记录
打开记录集后,就可以往里面添加记录了,添加一条记录一般按如下步骤进行。
o 调用 Recordset 对象的 AddNew()方法新增一个空记录。
o 调用 Recordset 对象的 PutCollect()方法输入每个字段的值。
o 调用 Recordset 对象的 Update()方法更新数据库中的数据。
下面是添加记录的一段代码:(假设 m_ID 、m_Name 、m_Telephone 和 m_Address 均为
已经赋值的变量)
m_Recordset…》AddNew();
m_Recordset…》PutCollect(〃ID〃; m_ID);
m_Recordset…》PutCollect(〃Name〃; _variant_t(m_Name));
m_Recordset…》PutCollect(〃Telephone〃; m_Telephone);
m_Recordset…》PutCollect(〃Address〃; _variant_t(m_Address));
m_Recordset…》Update();
2 .修改记录
修改一条记录通常按如下步骤进行。
o 调用 Recordset 对象的 PutCollect()方法给当前记录的每个字段赋值。
o 调用 Recordset 对象的 Update()方法更新数据库中的数据。
以下是修改当前记录的代码:
m_Recordset…》PutCollect(〃ID〃; m_NewID);
·354 ·
…………………………………………………………Page 366……………………………………………………………
第 12 章 数据库开发
m_Recordset…》PutCollect(〃Name〃; _variant_t(m_NewName));
m_Recordset…》PutCollect(〃Telephone〃; m_NewTelephone);
m_Recordset…》PutCollect(〃Address〃; _variant_t(m_NewAddress));
m_Recordset…》Update();
3 .删除记录
利用 Recordset 对象的 Delete()方法很容易删除记录。Delete()方法的原型如下:
Delete(enum AffectedEnum AffectRecords);
参数 AffectRecords 是一个枚举型变量,用于指定删除方式。如果为此变量赋值
adAffectCurrent,则只有当前记录被删除;如果赋值为 adAffectedGroup,则符合过滤器的所
有行都被删除。下面一段代码将当前记录删除:
m_Recordset…》Delete(asAffectCurrent);
m_Recordset…》Update();
12。6。5 关闭记录集
当所有操作结束后,应当关闭记录集及当前连接,分别调用 Recordset 对象和 Connection
对象的 Close()方法即可,代码如下:
//关闭记录集
if (m_recordSet != NULL)
m_recordSet…》Close();
//关闭连接
if (m_connection !=NULL)
m_connection…》Close();
12。7 利用 ADO 查询并操作数据库实例
实例 12…2:ADO 使用实例。源代码在光盘中“12实例 12…2AdoTest”目录下。
前面讲了很多 ADO 的基础知识,读者也应该对 ADO 有了一个基本的了解,并且知道了
ADO 的常用函数和操作数据库的步骤。下面即将进行实际操作,这里将利用上面所讲的知识
编写一个程序,达到查询和操作数据库的目的。
在具体编写程序前,还要了解一下经常用到的两种类型,即_variant_t 和_bstr_t 。它们是
两个类,分别继承了 VARIANT 和 BSTR ,并增加了一些方法。之所以引入这两种数据类型,
是因为 必须设计成跨平台,在 里用不了 CString 类,这时需要有一种更普遍的方
式来处理字符串以及其他数据。VARIANT 就是一个巨大的 union ,包含了除 char * 以外的所
有的数据类型,BSTR 取代了 char* 。利用新引入的这两种类型,会使编程更加方便。比如从
结果集 rs 中读取数据,然后放到 List Box 控件 m_List 中。如果采用新的类型,具体代码如
下:
·355 ·
…………………………………………………………Page 367……………………………………………………………
Visual C++ 6。0 程序设计从入门到精通
_variant_t Holder;
Holder = rs…》GetCollect(〃FIELD_1〃);
m_List。AddString((char*)_bstr_t(Holder));
对比一下没有采用_variant_t 和 _bstr_t 的代码:
COleVariant covFieldValuel;
VARIANT vFieldValue;
CString Holder;
Rs…》GetFieldValue(〃FIELD_1〃; covFieldValue);
vFieldValue = (LPVARIANT)covFieldValue;
Holder。Format(〃%s〃;vFieldValue…》pbVal);
m_List。AddString(Holder);
另外在编写程序时要用 try 和 catch,否则 ADO 调用错误有可能使程序崩溃,一定要随
时记得捕捉__error 例外以及其他错误。
此外,在第 2 节介绍如何设置 ODBC 数据源时提到了利用 组件 DataSourceLocator ,
用户可以在程序运行的过程中动态设置数据源 。至于如何利用此组件,将在程序中详细说明。
下面具体介绍该程序的编写过程。
1.创建工程
新建一个 MFC 工程,取名为 AdoTest ,在第一步中选择单文档,在最后一步的基类选择
CFormView,分别如图 12…20 和图 12…21 所示。
图 12…20 单文档工程 图 12…21 基类选择 CFormView 类
2 .链入 ADO 库文件
在文件“stdafx。h ”中加入如下语句:
#import 〃C:Program Filesmon Filessystemole dboledb32。dll〃 no_namespace
#import 〃c:program filesmon filessystemadomsado15。dll〃 no_namespace rename(〃EOF〃; 〃adoEOF〃)
加入第二个语句前面已经讲过原因,加入第一个语句是由于在程序中将要采用 组
件 DataSourceLocator 。这里需要注意一点,两个 import 语句中都用了 no_namespace ,当引入
·356 ·
…………………………………………………………Page 368……………………………………………………………
第 12 章 数据库开发
多个库时,可能会引起名字的冲突。不过由于本程序较小,且只有两个库,所以在这里用
no_namespace 也可以。
3 .界面设计
在显示数据库内容时,可以利用 ActiveX 控件 DataGrid Control ,将会大大简化编码的工
Project ” Add To Project ” ponents
作。至于如何做,下面的程序中进行介绍 。通过单击“ “ “
and Controls ”,弹出如图 12…22 所示的对话框,在此对话框中打开“Registered ActiveX Controls ”
文件夹,然后选中“Microsoft DataGrid Control; Version 6。0(OLEDB) ”,如图 12…23 所示。单
击“Insert ”按钮,在弹出的确认对话框中单击“确定”按钮,然后在弹出的“Confirm Classes ”
对话框中取默认值,如图 12…24 所示,单击“OK ”按钮返回“ponents and Controls Gallery ”
对话框,单击“Close ”按钮即可将 DataGrid Control 加入到工程中。
图 12…22 添加组件及控件对话框
图 12…23 添加 DataGrid 控件对话框
接 下 来 进 行 界 面 设 计 , 在 资 源 视 图 中 找 到 对 话 框 文 件 夹 , 然 后 在 ID 为
IDD_ADOTEST_FORM 的对话框中添加一些控件,如图 12…25 所示。其中,最左边是一个
List Box 控件,用来显示数据库中的表项 ;中间就是上面所加入的 DataGrid ActiveX 控件(在
控件栏的最下端可以看到一个红色的标志,这就是 DataGrid Control );最右边的 5 个按钮
·357 ·
…………………………………………………………Page 369……………………………………………………………
Visual C++ 6。0 程序设计从入门到精通
Button1~Button5 分别对应设定数据源、连接数据库、执行 SQL 语句、断开连接和退出;对
话框下面的“连接串”和“操作”都是 Static Text 控件;两个 Edit Box 控件分别用来显示连
接语句和输入 SQL 语句。
图 12…24 类确认对话框
图 12…25 添加控件后的对话框
然后对各控件的属性进行设置,具体设置如表 12…24 所示。
表 12…24 控件属性设置表
控件 ID Caption
Datagrid 控件 IDC_DATAGRID DataGrid1
List Box 控件 IDC_LIST
Edit 控件 1 IDC_CONNECTION_STRING
Edit 控件 2 IDC_EXECUTE_STRING
按钮控件 1 IDC_SOURCE 数据源
按钮控件 2 IDC_CONNECT 连接
按钮控件 3 IDC_EXECUTE 操作
按钮控件 4 IDC_DISCONNECT 断开
按钮控件 5 IDC_QUIT 退出
·358 ·
…………………………………………………………Page 370……………………………………………………………
第 12 章 数据库开发
4 .编写代码
在正式开始编写代码之前,需要进一步明确此程序要实现的功能。用户单击“数据源”
按钮,则可以利用 组件 DataSourceLocator 动态设置数据源,同时在 Edit 控件 1 中将连
接串显示出来 。设定数据源后,用户单击“连接”按钮,程序连接到数据源,同时在 List Box
控件中将数据库中所有表名都显示出来。然后用户可以在 Edit 控件 2 中输入 SQL 语句,单
击“操作”按钮对数据库进行操作 。当操作完毕后,单击“断开”按钮断开与数据源的连接。
最后单击“退出”按钮退出程序界面。同时为了方便用户,程序增加了两个功能。当用户在
List Box 控件中单击某个表名时,在 DataGrid 控件中会相应的显示出此表的所有记录;当用
户在 DataGrid 控件中表的某列单击时,此列按照升序进行排列,再次单击此列,则降序排列。
注意:为了让程序更加简洁,本节在介绍操作数据库时,没有利用 12。6。4 小节介绍的对数据库记
录进行操作的方法,而是直接用 SQL 语句操作。读者也可以根据 12。6。4 小节所讲内容自己
编写操作记录的代码。
(1)初始化变量
首 先 在 文 件 “ CAdoTestView。h ” 的 开 头 加 入 语 句#include 〃datagrid。h〃 , 然 后 为 类
CAdoTestView 添加如表 12…25 所示的成员变量,它们都是 public 成员变量。
表 12…25 类 CAdoTestView 成员变量表
成员变量 功能
_ConnectionPtr m_Connection 连接数据源
_RecordsetPtr m_Recordset 打开记录集
CListBox m_ListBox 与控件 List Box 关联
CDataGrid m_DataGrid 与控件 DataGrid 关联
CString m_strConnection 连接串,与 Edit 控件 1 关联
CString m_strSQL 对数据库操作的 SQL 语句,与 Edit 控件 2 关联
CString m_strTableName 表名
BOOL isDesc 是否降序排列
然后在 CAdoTestView 类的构造函数中先进行一些初始化工作,代码如下:
CAdoTestView::CAdoTestView()
: CFormView(CAdoTestView::IDD)
{
//{{AFX_DATA_INIT(CAdoTestView)
// NOTE: the ClassWizard will add member initialization here
//}}AFX_DATA_INIT
// TODO: add construction code here
m_strConnection = _T(〃〃);
m_strSQL = _T(〃〃);
m_strTableName = _T(〃〃);
isDesc = FALSE;
}
最后要将各控件和它们对应的变量关联起来,代码如下:
void CAdoTestView::DoDataExchange(CDataExchange* pDX )
·359 ·
…………………………………………………………Page 371……………………………………………………………
Visual C++ 6。0 程序设计从入门到精通
{
CFormView::DoDataExchange(pDX);
//{{AFX_DATA_MAP(CAdoTestView)
// NOTE: the ClassWizard will add DDX and DDV calls here
//}}AFX_DATA_MAP
DDX_Control(pDX; IDC_LISTBOX; m_strListBox);
DDX_Control(pDX; IDC_DATAGRID; m_strDataGrid);
DDX_Text(pDX; IDC_CONNECTION_STRING; m_strConnection);
DDX_Text(pDX; IDC_EXECUTE_STRING; m_strSQL);
}
(2 )编写各消息响应函数
前面已经说过,在编写 ADO 的程序时,要用 try 和 catch,否则 ADO 调用错误有可能使
程序崩溃,一定要随时记得捕捉__error 例外以及其他错误。因此首先编写捕获例外时的
处理函数。为类 CAdoTestView 添加 public 成员函数 void GenerateError(HRESULT hr; PWSTR
pwszDescription) ,代码如下:
void CAdoTestView::GenerateError(HRESULT hr; PWSTR pwszDescription )
{
CString m_strError;
m_strError。Format(〃Run…time error ’%d (%x)’〃; hr; hr);
m_strError += 〃nr〃;
m_strError += pwszDescription;
AfxMessageBox(m_strError);
}
接下来为各个按钮添加响应函数。首先为用来设定数据源的按钮添加响应函数:打开
“ClassWizard ”对话框,在“Class name ”下拉菜单中选择 CAdoTestView,在 Object IDs 下
拉列表中选择 IDC_SOURCE ,在“Messages ”下拉列表中选择 BN_CLICKED ,单击“Add
Function ”,采用默认的函数名,然后依次单击“OK ”和“Edit Code ”按钮,定位到函数
CAdoTestView::OnSource(),代码如下:
void CAdoTestView::OnSource()
{
// TODO: Add your control notification handler code here
// TODO: Add your control notification handler code here
HRESULT hr;
IDataSourceLocatorPtr m_dlPrompt = NULL;
_ConnectionPtr m_Conn = NULL;
//初始化
::CoInitialize(NULL);
·360 ·
…………………………………………………………Page 372……………………………………………………………
第 12 章 数据库开发
//创建 IDataSourceLocatorPtr 的实例
hr = m_dlPrompt。CreateInstance(__uuidof(DataLinks));
//弹出数据连接的对话框
m_Conn = m_dlPrompt…》PromptNew();
if (m_Conn!=NULL)
{
//将连接字符串复制到 m_strConnect 中
m_strConnection。Format(〃%s〃; (char*)m_Conn…》ConnectionString);
//将变量中的值保存到控件中
快捷操作: 按键盘上方向键 ← 或 → 可快速上下翻页 按键盘上的 Enter 键可回到本书目录页 按键盘上方向键 ↑ 可回到本页顶部!
温馨提示: 温看小说的同时发表评论,说出自己的看法和其它小伙伴们分享也不错哦!发表书评还可以获得积分和经验奖励,认真写原创书评 被采纳为精评可以获得大量金币、积分和经验奖励哦!