/***********************ADO接口简介*********************************/
ADO库包含三个基本接口:_ConnectionPtr接口、_CommandPtr接口和_RecordsetPtr接口。

_ConnectionPtr接口返回一个记录集或一个空指针。
通常使用它来创建一个数据连接或执行一条不返回任何结果的SQL语句,如一个存储过程。
使用_ConnectionPtr接口返回一个记录集不是一个好的使用方法。
通常同Cdatabase一样,使用它创建一个数据连接,然后使用其它对象执行数据输入输出操作。

_CommandPtr接口返回一个记录集。
它提供了一种简单的方法来执行返回记录集的存储过程和SQL语句。
在使用_CommandPtr接口时,你可以利用全局_ConnectionPtr接口,也可以在_CommandPtr接口里直接使用连接串。
如果你只执行一次或几次数据访问操作,后者是比较好的选择。
但如果你要频繁访问数据库,并要返回很多记录集,那么,你应该使用全局_ConnectionPtr接口创建一个数据连接,
然后使用_CommandPtr接口执行存储过程和SQL语句。

_RecordsetPtr是一个记录集对象。
与以上两种对象相比,它对记录集提供了更多的控制功能,如记录锁定,游标控制等
。同_CommandPtr接口一样,它不一定要使用一个已经创建的数据连接,
可以用一个连接串代替连接指针赋给_RecordsetPtr的connection成员变量,让它自己创建数据连接。
如果你要使用多个记录集,最好的方法是同Command对象一样使用已经创建了数据连接的全局_ConnectionPtr接口
,然后使用_RecordsetPtr执行存储过程和SQL语句。

/***********************基本流程***********************************/
(1)初始化COM库,引入ADO库定义文件
(2)用Connection对象连接数据库
(3)利用建立好的连接,通过Connection、Command对象执行SQL命令,或利用Recordset对象取得结果记
  录集进行查询、处理。
(4)使用完毕后关闭连接释放对象。

/***********************【1】COM库的初始化***********************************/
我们可以使用AfxOleInit()来初始化COM库,这项工作通常在CWinApp::InitInstance()的重载函数中完
成,请看如下代码:

BOOL CADOTest1App::InitInstance()
{
  AfxOleInit();
  ......
}

/*****************【2】用#import指令引入ADO类型库**********************/
我们在stdafx.h中加入如下语句:
#import "c:/program files/common files/system/ado/msado15.dll" no_namespace rename("EOF","adoEOF")
其最终作用同我们熟悉的#include类似,编译的时候系统会为我们生成msado15.tlh,ado15.tli两个C++头文件来定义ADO库。

/***************【3】创建Connection对象并连接数据库********************/

首先我们需要添加一个指向Connection对象的指针:

_ConnectionPtr m_pConnection;

BOOL CADOTest1Dlg::OnInitDialog()
{
CDialog::OnInitDialog();
try
{
  HRESULT hr = m_pConnection.CreateInstance("ADODB.Connection");//创建Connection对象
  if(SUCCEEDED(hr))
  {
  hr = m_pConnection->Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=test.mdb","","",adModeUnknown);//连接数据库
  //上面一句中连接字串中的Provider是针对ACCESS2000环境的,对于ACCESS97,需要改为:Provider=Microsoft.Jet.OLEDB.3.51;  }
  }
}
catch (_com_error e) //COM错误取得,当执行COM功能的时候,如果出错,可以捕捉到_com_error的异常

  CString strComError;
  strComError.Format("错误编号: %08lx/n错误信息: %s/n错误源: %s/n错误描述: %s",
      e.Error(),                  // 错误编号
      e.ErrorMessage(),          // 错误信息
      (LPCSTR) e.Source(),        // 错误源
      (LPCSTR) e.Description());  // 错误描述     
 
  ::MessageBox(NULL,strComError,"错误",MB_ICONEXCLAMATION);
}
}
也可以使用UDL文件进行连接。
try

m_pConnection.CreateInstance(__uuidof(Connection)); 
m_pConnection->ConnectionString ="File Name=e.udl";
m_pConnection->Open("","","",NULL);
}
catch(_com_error e)
{....}

◆在这段代码中我们是通过Connection对象的Open方法来进行连接数据库的,下面是该方法的原型

HRESULT Connection15::Open (_bstr_t ConnectionString, _bstr_t UserID, _bstr_t Password, long Options )

ConnectionString 为连接字串,
UserID    是用户名,
Password  是登陆密码,
Options    是连接选项,用于指定Connection对象对数据的更新许可权,
    Options可以是如下几个常量:

adModeUnknown:  缺省。当前的许可权未设置
  adModeRead:    只读
  adModeWrite:  只写
  adModeReadWrite:  可以读写
  adModeShareDenyRead: 阻止其它Connection对象以读权限打开连接
  adModeShareDenyWrite: 阻止其它Connection对象以写权限打开连接
  adModeShareExclusive: 阻止其它Connection对象打开连接
  adModeShareDenyNone: 允许其它程序或对象以任何权限建立连接

◆常用的数据库连接方法:

(1)通过JET数据库引擎对ACCESS2000数据库的连接
m_pConnection->Open("Provider=Microsoft.Jet.OLEDB.4.0;Data Source=C://test.mdb","","",adModeUnknown);

(2)通过DSN数据源对任何支持ODBC的数据库进行连接:
m_pConnection->Open("Data Source=adotest;UID=sa;PWD=;","","",adModeUnknown);
//m_pConnection->Open("DSN=test;","","",0);  //连接叫作test的ODBC数据源

(3)不通过DSN对SQL SERVER数据库进行连接:
m_pConnection->Open("driver={SQL Server};Server=127.0.0.1;DATABASE=vckbase;UID=sa;PWD=139","","",adModeUnknown);
其中Server是SQL服务器的名称,DATABASE是库的名称

◆先介绍Connection对象中两个有用的属性ConnectionTimeOut与State
ConnectionTimeOut用来设置连接的超时时间,需要在Open之前调用,例如:
  m_pConnection->ConnectionTimeout = 5; //设置超时时间为5秒
  m_pConnection->Open("Data Source=adotest;","","",adModeUnknown);

State属性指明当前Connection对象的状态,0表示关闭,1表示已经打开,我们可以通过
读取这个属性来作相应的处理,例如:
if(m_pConnection->State)
  m_pConnection->Close(); //如果已经打开了连接则关闭它

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执行完后返回一个指向记录集的指针,下面我们给出具体代码并作说明。 
try
{
_variant_t ra;

m_pConnection->Execute("CREATE TABLE student(Sno INTEGER,Sname TEXT,Ssex TEXT,Sage INTEGER,Sdept TEXT)",&ra,adCmdText);
m_pConnection->Execute("INSERT INTO 学生信息(学号,姓名,年龄,生日) VALUES (112105, '程红秀',22,'1982-08-16')",&ra,adCmdText);//往表格里面添加记录
m_pRecordset = m_pConnection->Execute("SELECT COUNT(*) FROM 学生信息",&ra,adCmdText);  //执行SQL统计命令得到包含记录条数的记录集

_variant_t vCount = m_pRecordset->GetCollect((_variant_t)(long)(0)); //取得第一个字段的值放入vCount变量

m_pRecordset->Close();
CString message;
message.Format("共有%d条记录",vCount.lVal);
AfxMessageBox(message); 
}

catch (_com_error e)
{ ...}

◆(2)利用 Command对象 来执行SQL命令
try
{
  _CommandPtr m_pCommand;
  m_pCommand.CreateInstance("ADODB.Command");
 
  m_pCommand->ActiveConnection = m_pConnection; //关键的一句,将建立的连接赋值给它

m_pCommand->CommandText="INSERT INTO 学生信息(学号,姓名,年龄,生日) VALUES (112105, '程红秀',22,'1982-08-16')";
  m_pCommand->Execute(NULL,NULL,adCmdText);

m_pCommand->CommandText="SELECT COUNT(*) FROM 学生信息";
  m_pRecordset=m_pCommand->Execute(NULL,NULL,adCmdText);

_variant_t vCount = m_pRecordset->GetCollect((_variant_t)(long)0); //取得第一个字段的值
 
  CString str;
  str.Format("共有%d条记录",vCount.lVal);
  AfxMessageBox(str);

m_pRecordset->Close();

}
catch (_com_error e) {...}

在这段代码中我们只是用Command对象来执行了SELECT查询语句,
Command对象在进行存储过程的调用中能真正体现它的作用。下次我们将详细介绍。

◆(3)直接用Recordset对象进行查询取得记录集
例如
  m_pRecordset->Open("SELECT * FROM 学生信息",_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请参考本文中对Connection对象的Execute方法的介绍

/***********************【5】记录集的遍历、更新*************************/

根据我们刚才通过执行SQL命令建立好的 学生信息 表,它包含四个字段:学号,姓名,年龄,生日
以下的代码实现:打开记录集,遍历所有记录,删除第一条记录,添加三条记录,移动光标到第二条
记录,更改其年龄,保存到数据库。

try
{
  _variant_t vUsername,vBirthday,vID,vOld;
  _RecordsetPtr m_pRecordset;
 
  m_pRecordset.CreateInstance("ADODB.Recordset");
  m_pRecordset->Open("SELECT * FROM 学生信息",_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("姓名");  //取得姓名字段的值
  vOld = m_pRecordset->GetCollect("年龄");
  vBirthday = m_pRecordset->GetCollect("生日");
 
  TRACE("id:%d,姓名:%s,年龄:%d,生日:%s/r/n",
    vID.lVal,
    (LPCTSTR)(_bstr_t)vUsername,
    vOld.lVal,
    (LPCTSTR)(_bstr_t)vBirthday);  //在DEBUG方式下的OUTPUT窗口输出记录集中的记录

m_pRecordset->MoveNext();    //移到下一条记录
  }
 
  m_pRecordset->MoveFirst();    //移到首条记录
  m_pRecordset->Delete(adAffectCurrent);    //删除当前记录
 
 
  for(int i=0;i<3;i++)    //添加三条新记录并赋值
  {
  m_pRecordset->AddNew();    //添加新记录
  m_pRecordset->PutCollect("学号",_variant_t((long)(i+10)));
  m_pRecordset->PutCollect("姓名",_variant_t("王斌年"));
  m_pRecordset->PutCollect("年龄",_variant_t((long)21));
  m_pRecordset->PutCollect("生日",_variant_t("1930-3-15"));
  }
 
  m_pRecordset->Move(1,_variant_t((long)adBookmarkFirst));    //从第一条记录往下移动一条记录,即移动到第二条记录处
  m_pRecordset->PutCollect(_variant_t("年龄"),_variant_t((long)45));    //修改其年龄
  m_pRecordset->Update();    //保存到库中
 
} catch (_com_error e){}

/***********************【6】关闭记录集与连接 *************************/
记录集或连接都可以用Close方法来关闭     
m_pRecordset->Close(); //关闭记录集 
m_pConnection->Close(); //关闭连接

在stdafx.h中进行宏定义:
#if !defined CATCH_ERROR
#define CATCH_ERROR          /
  {            /
  CString strComError;      /
  strComError.Format("错误编号: %08lx/n错误信息: %s/n错误源: %s/n错误描述: %s", /
        e.Error(),                  /
        e.ErrorMessage(),          /
        (LPCSTR) e.Source(),        /
        (LPCSTR) e.Description());  /
  ::MessageBox(NULL,strComError,"错误",MB_ICONEXCLAMATION); /
  }
#endif
使用方法:
try
{  ...}
catch(_com_error e)
{
CATCH_ERROR;
}

*********ADO接口简介*********相关推荐

  1. 4-1 ADO.NET简介

    第四章 ADO.NET数据库访问技术 本章内容 4-1  ADO.NET 简介 4-2 ADO.NET-插入.删除.修改.检索数据 4-3 DataGridView 控件 - 显示和操作数据 4-4 ...

  2. Java Iterator 接口简介和简单用法.

    Iterator 的中文意思是迭代器. 单单从中文翻译也不易理解迭代器的意思啊,  其实Iterator是1个接口,  它的作用就是遍历容器的所有元素. 一, Iterator 接口简介 Iterat ...

  3. java中接口文件创建_功能接口简介–在Java 8中重新创建的概念

    java中接口文件创建 世界各地的所有Java开发人员都将至少使用以下接口之一:java.lang.Runnable,java.awt.event.ActionListener,java.util.C ...

  4. 功能接口简介–在Java 8中重新创建的概念

    世界各地的所有Java开发人员都将至少使用以下接口之一:java.lang.Runnable,java.awt.event.ActionListener,java.util.Comparator,ja ...

  5. 接口简介 java 1614100890

    接口简介 java 1614100890 生活中的接口 不同厂家的接口 家里的电器都可以使用 为什么? 因为他们是按照相同的规范设计的 不同厂家生产的电脑,USB接口也是同理 相关设置都可以直接使用 ...

  6. 以太网接口MII,RMII,SMII,GMII总线接口简介

    以太网接口MII,RMII,SMII,GMII总线接口简介 所有的这些接口都从MII而来,MII是(MediumIndependent Interface)的意思,是指不用考虑媒体是铜轴.光纤.电缆等 ...

  7. TypeScript基础入门 - 接口 - 简介

    转载地址 TypeScript基础入门 - 接口 - 简介 项目实践仓库 https://github.com/durban89/typescript_demo.git tag: 1.0.6 为了保证 ...

  8. 路由器广域网接口简介

    路由器广域网接口简介1.WAN侧FE接口WAN侧FE接口工作在网络层,可以配置IP地址,处理三层协议,提供路由功能,FE接口支持的最大速率为100Mbit/s2.WAN侧GE接口WAN侧GE接口工作在 ...

  9. keil 生成三角波dac0832_弹性波,时域显式接口简介

    COMSOL Multiphysics® 软件 5.5 版本中提供了一个节省内存的物理场接口,用于模拟弹性波在固体中的传播(结构中的振动).该弹性波,时域显式接口基于时域显示时间积分方案的高阶间断伽辽 ...

最新文章

  1. 【OpenCV 4开发详解】圆形检测
  2. Python找出序列中出现最多的元素
  3. leetcode算法题--分割等和子集★
  4. react 调用组件方法_React源码分析1 — 组件和对象的创建(createClass,createElement)...
  5. 8.4 parted:磁盘分区工具
  6. HP proliant服务器从usb启动
  7. centos下java编程工具_centos下安装JAVA开发工具(4)------Redis
  8. js打印插件_使用 Nodejs 开发一个 commitlint-release 插件
  9. 西门子g120变频器接线图_【直播】S71200与SINAMICS G120变频器的profinet通信
  10. 使用 net C 发送邮件(带成功案例)
  11. 怎么搜索计算机共享盘,怎样查找其他计算机上的共享文件
  12. OpenGL(五)立体图形
  13. vue-cropper图片裁剪 结合 OSS 实现图片直传(后台)
  14. 八进制换算成二进制、十进制、十六进制
  15. XML解析之DOM、SAX、JAXP、DOM4J
  16. 集成Health Kit时因证书问题出现错误码50063的解决方案
  17. 实现微信向多好友定时发送信息
  18. CAN301 移动计算 个人笔记
  19. 第1节、一个萝卜一个坑——计数排序
  20. WPF实现贪食蛇小案例

热门文章

  1. POI操作Excel设置前景色背景色
  2. 【总结】程序员运营公众号一年总结
  3. 【CSDN 竞赛第四期】参赛题解(JAVA)
  4. 写代码实现堆溢出、栈溢出、永久代溢出、直接内存溢出
  5. Java咖啡馆(2)——第一杯咖啡
  6. 践行社会责任 | 华云数据荣登2021新型实体企业百强榜 用创新技术服务实体经济
  7. 领导说“辛苦了”,千万别回“应该的”, 这是我听过最扎心的答案了!
  8. 新一代萌主!人见人爱的夏洛特小公主
  9. 继勒索病毒“永恒之蓝”445端口被封之后,在公网实现smb文件共享
  10. ADS1220 24位高精度ADC芯片驱动