使用ADO实现BLOB数据的存取

一、前言

在实际的开发过程中我们常常需要存储较大的二进制数据对象,比如:图像、音频文件、
或其它二进制数据,这些数据我们称之为二进制大对象BLOB(Binary Large Object),
其存取的方式与普通数据有所区别。本文将介绍利用ADO在数据库中存取BLOB数据的
具体实现过程,并给出实现图像存取显示的完整示例工程。

二、前期准备

首先我们建立一张名为userinfo的表,包含三个字段:id,username,old,photo,其中
photo是一个可以存储二进制数据的字段。

2.1 在SQL SERVER中我们可以在Query Analyzer中直接输入如下语句创建:

CREATE TABLE [dbo].[userphoto] (

[id] [int] IDENTITY (1, 1) NOT NULL ,

[username] [varchar] (50) NULL ,

[old] [int] NULL ,

[photo] [image] NULL

) ON [PRIMARY] TEXTIMAGE_ON [PRIMARY]

其中photo我们定义为image类型的字段。

2.2 在ACCESS中创建的方法如下:

建立一张新表包括id,username,old,photo四个字段,然后打开表,选视图菜单中设
计视图,将id设置为自动编号的递增长整型,username为文本,old为数字,photo
为OLE对象。

在我们的示例工程中已经包含了一个建立好的ACCESS2000的库,你可以直接拿来使
用。

三、具体步骤

3.1 BLOB数据的保存

BLOB类型的数据无法用普通的方式进行存储,我们需要使用AppendChunk函数,
AppendChunk包含在Field对象中,原型如下:

HRESULT AppendChunk (const _variant_t & Data );

从函数原型中可以看到关键的问题是我们需把二进制数据赋值给VARIANT类型的变
量,下面我们给出具体的代码并作简单的分析:

///假设m_pBMPBuffer指针指向一块长度为m_nFileLen的二进制数据,并且已经成功
打开了记录集对象m_pRecordset///

char *pBuf = m_pBMPBuffer;

VARIANT varBLOB;

SAFEARRAY *psa;

SAFEARRAYBOUND rgsabound[1];

m_pRecordset->AddNew(); ///
添加新记录

m_pRecordset->PutCollect("username",_variant_t("小李")); ///
为新记录填充username字段

m_pRecordset->PutCollect("old",_variant_t((long)28); ///
填充old字段

if(pBuf)

{

rgsabound[0].lLbound = 0;

rgsabound[0].cElements = m_nFileLen;

psa = SafeArrayCreate(VT_UI1, 1, rgsabound); ///
创建SAFEARRAY对象

for (long i = 0; i < (long)m_nFileLen; i++)

SafeArrayPutElement (psa, &i, pBuf++); ///
将pBuf指向的二进制数据保存到SAFEARRAY对象psa中

varBLOB.vt = VT_ARRAY | VT_UI1; ///
将varBLOB的类型设置为BYTE类型的数组

varBLOB.parray = psa; ///
为varBLOB变量赋值

m_pRecordset->GetFields()->GetItem("photo")->AppendChunk(varBLOB);///
加入BLOB类型的数据

}

m_pRecordset->Update(); ///
保存我们的数据到库中

至此我们的数据已经成功地保存到了数据库中,接下来我们所要做的工作便是将该
数据提取出来,让我们继续吧!

3.2 BLOB数据的读取

对应于保存数据时我们所使用的AppendChunk函数,读取数据应该使用GetChunk函

数,GetChunk的原型如下:

_variant_t GetChunk (long Length );

给出数据的长度后GetChunk将返回包含数据的VARIANT类型变量,然后我们可以利用
SafeArrayAccessData函数得到VARIANT变量中指向数据的char *类型的指针,以方
便我们的处理,具体代码如下:

long lDataSize = 
m_pRecordset->GetFields()->GetItem("photo")->ActualSize;///得到数据的长

if(lDataSize > 0)

{

_variant_t varBLOB;

varBLOB = 
m_pRecordset->GetFields()->GetItem("photo")->GetChunk(lDataSize);

if(varBLOB.vt == (VT_ARRAY | VT_UI1)) 
///判断数据类型是否正确

{

char *pBuf = NULL;

SafeArrayAccessData(varBLOB.parray,(void **)&pBuf); 
///得到指向数据的指针

/*****在这里我们可以对pBuf中的数据进行处理*****/

SafeArrayUnaccessData (varBLOB.parray);

}

}

以上我们成功实现了BLOB数据在数据库中的存取,为了让大家有现成的例子可以参
考,本文提供了示例工程,在示例工程中我们在数据库中保存图像数据,并可以对
这些图像进行浏览、修改,该例子还涉及到如何用char *指向的BMP文件数据创建
BITMAP对象然后显示出来。

【转】MFC下用ADO连接SQL SERVER,保存图片,BLOB相关推荐

  1. MFC开发之使用ADO连接SQL Server

    "观千剑而后视器,操千曲而后晓声."--<文心雕龙> 最近在学C++一个框架--MFC,犯难了,搞了一周,阅读了不下百篇博主的文章,还询问了老师,重装了VS(甚至201 ...

  2. VC++使用ADO连接SQL Server数据库

    基本步骤: 1.创建一个基于对话框的应用程序ADODatabase. 2.创建一个用来连接的ADO类库.系统菜单中Insert->New Class项,class type选择Generic C ...

  3. LabVIEW通过ADO连接SQL Server 方法二

    数据库连接方法2 一.数据源配置 1. 打开ODBC数据源管理器,点击"添加" 2. 选择SQL Server,点击"完成" 3. 输入名称,并选择服务器,点击 ...

  4. MFC使用ADO连接SQL SERVER数据库实现的高校教材管理系统

    摘 要 随着高校规模的扩大和教学的改革的深入,高校的教学水平和管理在稳步提高,而高校的教材管理环节起着为教学和科研提供软环境的重要作用,是与高校综合能力的增强相辅而成的.而现有的高校教材管理系统大多还 ...

  5. vc通过ADO连接sql server 2000的核心代码

    建立一个mfc对话框工程,一切都默认然后完成,接下照我的方法做吧,一定可以的! 注: ***** 代表你所建立的工程名称 先在stdafx.h中导入#import  " c:\program ...

  6. 网络环境下连接SQL Server和Oracle 19c的方法

    网络环境下连接SQL Server和Oracle 19c的方法 1.SQL Server 2019连接方法 1.1 启动SQL Server 2019配置管理器 因为软件都是64位的,从SQL Ser ...

  7. Linux下Oracle19C RAC通过Gateways(透明网关)连接Sql server数据库

    Linux下Oracle RAC通过Gateways(透明网关)连接Sql server数据库 文章目录 注意事项: 实验环境: 软件包下载: 操作步骤: 1.传输Gateways压缩包并解压得到ga ...

  8. linux下qt的odbc库编译,linux qt QODBC连接sql server 驱动 有关问题 ,实在是没辙了

    linux qt QODBC连接sql server 驱动 问题 ,实在是没辙了 程序在linux下,现在需要连接windows的sqlserver . db = QSqlDatabase::data ...

  9. ADO.NET连接SQL Server数据库

    在.NET应用程序中,创建数据连接分为三步: 一.定义连接字符串 不同的数据库连接字符串的格式不同,一般都包括要连接的数据库提供都名称.登陆信息以及要使用的数据库名称. 注意:在定义连接字符串之前,一 ...

最新文章

  1. vue 实例化几种方式_Vue组件的三种调用方式
  2. Symantec Backup Exec System Recovery简明安装手册
  3. 基于C++实现五子棋AI算法思想
  4. 开源音乐播放器_测试4个开源音乐播放器等
  5. 我为什么从外包公司离职了?
  6. ASR6505基于STM 8位MCU与SX1262 的SiP全频段LoRa芯片
  7. 一起学习log4cxx
  8. / ./ ../的区别
  9. 手机软件的测试主要有哪些方面的测试,性能测试用什么去测试好
  10. python多线程破解压缩包_我用 Python 破解了同事的加密压缩包!
  11. linux igmp 属于那层协议,igmp协议属于哪一层
  12. Oracle hint认识
  13. 总结一些网站加密和混淆技术
  14. 如何解决NavigationDuplicated: Avoided redundant navigation to current location:问题
  15. 【人工智能项目】ImageNet数据集介绍以及数字图像处理技术
  16. 机器人周志_唐骏:机器人时代已如梦初醒
  17. 管理信息系统【五】之 系统分析
  18. Complete Internet Repair(电脑网络修复工具)官方中文版V8.1.3.5222 | 网络修复工具使用后仍不能上网?快试试万能网络修复大师
  19. 【CSS】如何实现价格文字中间划一条线
  20. 电能表校验装置TD1550直流电能表检定装置

热门文章

  1. 优秀设计师必备:视觉传达设计与视觉思维
  2. Docker容器的文件系统管理
  3. java s0 s1_业余草告诉你Java GC 变量含义(S0 S1 E O P YGC YGCT FGC FGCT G
  4. hsi转rgb公式matlab,rgb与hsi模型转换matlab程序
  5. java.text._Java.text
  6. Failed to execute goal org.apache.maven.plugins:maven-surefire-plugin:2.12.4:test (default-test)
  7. Otter 异地机房数据同步的demo实施
  8. Java_01_环境变量的配置
  9. Java Map hashCode深究
  10. 测试常用shell语句——数值,数组类型;函数创建