在VC下采用ADO实现BLOB(Binary)数据的存储,读取,修改,删除。

作者:邵盛松 2009-09-05

前言

1关于的BLOB(Binary)数据的存储和读取功能主要参考了MSDN上的一篇《AppendChunk and GetChunk Methods Example (VC++)》,原文地址是http://msdn.microsoft.com/en-us/library/ms807920.aspx。还有www.vckbase.com上有一篇文章《使用ADO实现BLOB数据的存取 -- ADO开发实践之二》,原文地址是http://www.vckbase.com/document/viewdoc/?id=252

我在这篇博文当中增加了对BLOB(Binary)数据的存储,读取代码的分析。

2关于BLOB(Binary)数据的修改,我试验过多种方法,这个问题也花了比较长的时间才得到解决,原来的方法是从文件读取的数据,将更改的数据转化为CString类型,然后采用SQL的UPDATE语句进行数据的更新。后来在CSDN论坛上得到了vieri_ch的帮助,问题得到了比较好的一种解决方案。

实现方法:

1  BLOB(Binary)数据的存储,将一个文件存储到数据库。

存储部分代码分析写的比较详细,因为这里面得一些COM 结构和API需要说明一下

m_pRs为_RecordsetPtr类型

CFile   fileAdd;

if(fileAdd.Open(要存储文件的路径,CFile::modeRead)==0)    //打开文件

return;

_variant_t   varChunk;

long   nLength   =   fileAdd.GetLength();

BYTE*   pbuf;

pbuf   =   new   BYTE[nLength];

if(pbuf==NULL)

return;

fileAdd.Read(pbuf,nLength);

BYTE   *pBufEx;

pBufEx   =   pbuf;

SAFEARRAY*   psa;

SAFEARRAYBOUND   rgsabound[1];

rgsabound[0].lLbound   =   0;

rgsabound[0].cElements   =   nLength;

psa   =   SafeArrayCreate(VT_UI1, 1, rgsabound);

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

SafeArrayPutElement(psa,&i, pBufEx++);

VARIANT   varBLOB;

varBLOB.vt   =   VT_ARRAY   |   VT_UI1;

varBLOB.parray   =   psa;

m_pRs->ADOFields->GetItem(_variant_t("BLOB类型字段的名称"))->AppendChunk(varBLOB);

m_pRs->Update();

m_pRs->Close();

在代码中有一个SAFEARRAY和SAFEARRAYBOUND类型,该类型在头文件OAIdl.h中定义

typedef struct tagSAFEARRAY

{

USHORT cDims;

USHORT fFeatures;

ULONG cbElements;

ULONG cLocks;

SAFEARRAYBOUND rgsabound[ 1 ];

}SAFEARRAY;

SAFEARRAY结构体中包含了SAFEARRAYBOUND类型。

先说SAFEARRAYBOUND结构

cElements表示数组中元素的个数。

lLbound表示数组的下界。

代码中SAFEARRAYBOUND元素个数为读取文件的大小,下界是0

操作SAFEARRAY类型是由COM提供了一套API来处理的。oleauto.h 头文件中可以看到很多关于操作SAFEARRAY数据类型的API.以下少列出几个

WINOLEAUTAPI SafeArrayAllocDescriptor(UINT cDims, SAFEARRAY ** ppsaOut);

WINOLEAUTAPI SafeArrayAllocDescriptorEx(VARTYPE vt, UINT cDims, SAFEARRAY ** ppsaOut);

WINOLEAUTAPI SafeArrayAllocData(SAFEARRAY * psa);

WINOLEAUTAPI_(SAFEARRAY *) SafeArrayCreate(VARTYPE vt, UINT cDims, SAFEARRAYBOUND * rgsabound);

SAFEARRAY为什么会称为安全数组?

从名字上看是安全数组的意思。比如我们第一个数组,数组的大小为10,数组的下界是从0开始,这时我们访问下界为10的元素,这时就发生了错误,以为数组的下界最大为9。SAFEARRAY这样定义结构的作用是实现COM API函数可以限制我们访问下界为10的元素和其他一些安全操作数组的行为等

SAFEARRAY结构说明

成员 描述

cDims 数组的维数

fFeatures 标志

cbElements 数组元素的大小

cLocks 一个计数器,用来跟踪该数组被锁定的次数

pvData  指向数据缓冲的指针

标志表示是在堆上创建还是在栈上创建等还包括其他一些标志

FADF_AUTO 0x0001

在栈上创建数组

FADF_STATIC 0x0002

在堆上创建数组

FADF_EMBEDDED 0x0004

在结构中创建

FADF_FIXEDSIZE 0x0010

不能改变数组大小

FADF_RECORD 0x0020

记录容器

FADF_HAVEIID 0x0040

有IID 身份标记 数组

FADF_HAVEVARTYPE 0x0080

VT 类型数组

FADF_BSTR 0x0100

BSTR数组

FADF_UNKNOWN 0x0200

IUnknown* 数组

FADF_DISPATCH 0x0400

IDispatch* 数组

FADF_VARIANT 0x0800

VARIANTs数组

FADF_RESERVED 0xF0E8

余留,将来使用

COM API函数说明SafeArrayCreate创建数组。参数VT_UI1是表示unsigned int 1字节整数(BYTE)数组

vt是数组类型、lLbound是数组下界值(最小下标)和数组长度

SafeArrayPutElement函数是向一个安全数组中添加一个值,代码中采用循环一个一个得添加

ADO方法

AppendChunk将数据追加到大型文本、二进制数据 Field 或 Parameter 对象。

语法

object.AppendChunk Data

参数

object   Field 或 Parameter 对象

Data  变体型,包含追加到对象中的数据。

2  BLOB(Binary)数据的读取

将数据库中的数据读出并写入文件。

long   lDataLength = m_pRs->GetADOFields()->GetItem(_variant_t("BLOB类型的字段"))->ActualSize;

if (lDataLength>0)

{

_variant_t   varBLOB;

varBLOB=m_pRs->GetADOFields()->GetItem(_variant_t("BLOB类型的字段"))->GetChunk(lDataLength);

if(varBLOB.vt== (VT_ARRAY|VT_UI1) && varBLOB.vt!=VT_EMPTY && varBLOB.vt!=VT_NULL )

{

BYTE   *pBuf   =   NULL;

pBuf   =   (BYTE*)GlobalAlloc(GMEM_FIXED,lDataLength);

SafeArrayAccessData(varBLOB.parray,(void   **)&pBuf);

strFileName表示生成包含文件名的路径

CFile   outFile(strFileName,CFile::modeCreate|CFile::modeWrite); //构造新文件,如果文件存在,则长度变为0

outFile.Write(pBuf,lDataLength);

outFile.Close();

SafeArrayUnaccessData (varBLOB.parray);

}

}

m_pRs->Close();

3  BLOB(Binary)数据的修改

CFileException eFile;

CFile   fileSave;

strPath表示需要修改文件的路径

fileSave.Open(strPath,CFile::modeReadWrite|CFile::shareDenyWrite|CFile::shareDenyRead|CFile::typeBinary,&eFile);

_variant_t   varChunk;

long   nLength   =    fileSave.GetLength();

BYTE*   pbuf;

pbuf   =   new   BYTE[nLength];

if(pbuf   == NULL) return;

fileSave.Read(pbuf,nLength);

fileSave.Close();

BYTE   *pBufEx;

pBufEx   =   pbuf;

SAFEARRAY*   psa;

SAFEARRAYBOUND   rgsabound[1];

rgsabound[0].lLbound   =   0;

rgsabound[0].cElements   =   nLength;

psa   =   SafeArrayCreate(VT_UI1, 1, rgsabound);

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

SafeArrayPutElement(psa,&i, pBufEx++);

VARIANT   varBLOB;

varBLOB.vt   =   VT_ARRAY   |   VT_UI1;

varBLOB.parray   =   psa;

CString strSQL;

strSQL.Format(_T("UPDATE 表名 SET BLOB字段名=? WHERE ID=‘1‘");

m_pCom.CreateInstance(__uuidof(Command));

m_pCom->ActiveConnection = m_pCon;

m_pCom->CommandText = _bstr_t(strSQL);

m_pCom->CommandType = adCmdText;

m_pCom->Parameters->Append(m_pCom->CreateParameter(_T("@参数名"),adVarBinary, adParamInput, -1, varBLOB));

m_pCom->Execute(NULL,NULL,adCmdText);

4 BLOB(Binary)数据的修改

 

删除比较简单一点,同时表定义该字段允许为空

执行一个SQL语句就可以了

UPDATE 表名 SET BLOB字段名 = NULL WHERE ID=1

以上程序已经在Visual C++2005 Unicode下调试通过

时间: 2024-08-29 15:16:02

在VC下采用ADO实现BLOB(Binary)数据的存储,读取,修改,删除。的相关文章

VC++下封装ADO类以及使用方法

操作系统:windows 7软件环境:visual studio 2008 .Microsoft SQL 2005本次目的:介绍一个已经封装的ADO类,简单说明怎么导入使用 首先声明一下,这个封装的ADO类是在[vc知识库 ]下载的,因为最近在数据库课程设计,所以对vc++下使用ADO连接数据库不是很了解,故若是本文有错误的地方的,请不吝指出.具体的ADO类各个封装功能请进入[原文地址 ]查看.在此仅介绍使用此ADO封装类的入门,如连接数据库,显示记录等.一个测试例子如下: 使用步骤:1.先从[

使用ADO实现BLOB数据的存取 -- ADO开发实践之二

使用ADO实现BLOB数据的存取 -- ADO开发实践之二 http://www.360doc.com/content/11/0113/16/4780948_86256633.shtml 一.前言 在上一篇文章<>中我们详细介绍了ADO基本的操作方法,在实际的开发过程中我们常常需要存储较大的二进制数据对象,比如:图像.音频文件.或其它二进制数据,这些数据我们称之为二进制大对象BLOB(Binary Large Object),其存取的方式与普通数据有所区别.本文将介绍利用ADO在数据库中存取B

在VC下显示JPEG、GIF格式图像的一种简便方法

在VC下显示JPEG.GIF格式图像的一种简便方法 一. 引言  JPEG图像压缩标准随然是一种有损图像压缩标准,但由于人眼视觉的不敏感,经压缩后的画质基本没有发生变化,很快便以较高的压缩率得到了广泛的认可.GIF格式虽然仅支持256色但它对于颜色较少的图像有着很高的压缩率,甚至超过JPEG标准,也得到了广泛的认同.但作为众多程序员的一个重要的开发工具--Microsoft Visual C++ 6.0的MFC库却仅对没有经过任何压缩的BMP位图文件有着良好的支持,可以读取.显示.存储甚至在内存

[转]Net 下采用GET/POST/SOAP方式动态调用WebService C#实现

本文转自:http://www.cnblogs.com/splendidme/archive/2011/10/05/2199501.html 一直以来,我们都为动态调用WebService方法而烦恼.在.Net环境下,最常用的方法就是采用代理类来调用WebService,可以通过改变代理类的Url属性来实现动态调用,但当xmlns改变时就会出错,似乎要重新绑定Webservice并重新编译后才能再次运行.我无意中通过百度搜索找了一个采用GET/POST/SOAP方式动态调用WebService的

VC++下编译 程序“减肥”

在vc6 和 vs 2008下 编译 以下代码,不更改任何编译设置(vc6  40k , s2008 7k). 一.vc6下,Release 模式 编译处理. 1.去掉不必要的 链接库  工程(Project)-->设置(Settings)-->链接(link)属性页-->对象库/模块(object/library modules) 去掉所有的lib. 选择使用 MSVCRT.LIB kernel32.lib user32.lib. 可以忽略不必要的警告,比如 LINK:warning

VC下调试内存泄漏的办法

VC的控制台程序不会跟踪内存泄漏,如果需要调试控制台程序的内存情况,需要_CrtSetDbgFlag 详细原理我也不太懂,我也是从网上摘过来的,直接帖代码 // 内存泄漏跟踪 #ifdef WIN32 #include <SDKDDKVer.h> #ifdef _DEBUG #define _CRTDBG_MAP_ALLOC #include <stdlib.h> #include <crtdbg.h> #define VC_MemLeakCheck() _CrtSet

VC下&#39;long&#39; followed by &#39;long&#39; is illegal

long long 类型在VC6.0编译错误 long long int 是GCC(GNU C)的64位表示法. VC里面只能用__int64 有两种解决办法: 1.只需要把long long 替换成__int64即可, 注意一定要点击下保存按钮否则头文件修改无效还是编译不通过 2.或者是:如下 stdint.h文件重新建立,并输入以下信息 /* ISO C9x 7.18 Integer types <stdint.h> * Based on ISO/IEC SC22/WG14 9899 Co

VS2005环境下采用makefile编译、使用libjpeg.lib函数库

1.从www.ijg.org下载源码,解压后得到文件夹jpeg-8d 2.在文件夹里新建jconfig.h文件,将jconfig.vc里的内容拷到jconfig.h中 3.编译. Run->CMD->进入到C:\Program Files\Microsoft Visual Studio 8\VC\bin目录(不同的VC编译器目录也不一样)下 ->运行vcvars32.bat文件http://www.playerflash.org/,此时 这里的先进入makfefile.vc目录下 --&

EF6.0下建立ADO.NET实体数据模型错误解决方法

最近在学习EF+MVC的过程中遇到个很奇怪的问题:EF5.0下建立ADO.NET实体数据模型可以成功,可在EF6.0下建立ADO.NET实体数据模型就报错了,下图为错误详情: 在自己的电脑里压根就找不到这个这个路径,对于EF的引用也不是在这个目录下,不懂怎么就会出现这个路径. 在网上搜了很久都没找到中文的相关资料,英文的倒是找到一篇,连接:http://forums.asp.net/t/1949811.aspx?Error+building+an+entity+model+tt+is+empty