1 、引言
在許多項目開發(fā)中,由于存在著大量的數(shù)據(jù)需要存儲,通常采用數(shù)據(jù)庫來存儲這些數(shù)據(jù),使得數(shù)據(jù)庫技術(shù)被越來越廣泛地應(yīng)用。在這些數(shù)據(jù)中有簡單的文本文件,只需通過api的簡單編程即可實現(xiàn),但也存在許多大對象數(shù)據(jù),比如圖像、音頻和視頻等,其存儲形式有所不同。作為大對象數(shù)據(jù)的圖像數(shù)據(jù),一般以blob(二進(jìn)制大對象,binary large object)形式存儲,例如access數(shù)據(jù)庫中ole對象數(shù)據(jù)類型、sql server數(shù)據(jù)庫中的image數(shù)據(jù)類型等。在vc++ 7.0的環(huán)境下,為了高效快捷的訪問圖像數(shù)據(jù),需要采用合適的數(shù)據(jù)庫訪問技術(shù)。
在vc++中標(biāo)準(zhǔn)的訪問數(shù)據(jù)庫技術(shù)有很多種,根據(jù)開發(fā)的項目,采用了數(shù)據(jù)庫技術(shù)中最常見有效的兩種技術(shù)ado技術(shù)和mfc odbc在access數(shù)據(jù)庫中訪問圖像數(shù)據(jù)。
2、 圖像數(shù)據(jù)訪問原理
雖然圖像數(shù)據(jù)的格式有多種,例如bmp、jpg和gif等格式,但都可以看成是二進(jìn)制流,即blob類型。blob類型數(shù)據(jù)需要占用較大的硬盤空間和內(nèi)存,對存儲效率和查詢速度都有很大的影響。
圖像數(shù)據(jù)的訪問分存儲和讀取兩個過程。在存儲過程中,由于存放圖像數(shù)據(jù)的字段是可變長度(長度一般為0~2g)的blob類型,無法直接存儲,必須首先將存儲在文件中的圖像數(shù)據(jù)以二進(jìn)制流的形式讀到緩沖區(qū)中,然后再將緩沖區(qū)中的圖像數(shù)據(jù)添加到access數(shù)據(jù)庫中的ole對象中。在讀取過程中,將存放在access數(shù)據(jù)庫中的圖像數(shù)據(jù)讀到緩沖區(qū)中,然后在顯示在用戶界面上。圖像數(shù)據(jù)的訪問原理如圖1所示。
圖1 圖像數(shù)據(jù)訪問原理圖
3、 應(yīng)用ado訪問圖像數(shù)據(jù)
3.1 ado的介紹
ado(activex data objects)是microsoft數(shù)據(jù)庫應(yīng)用程序開發(fā)的接口,是建立在ole db技術(shù)之上的高層數(shù)據(jù)庫訪技術(shù),簡化了編程,且有利于程序的可移植性及可擴充性。
ado包括connection對象、command對象、parameter對象、recordset對象、field對象、error對象、property對象以及相應(yīng)的集合對象,這些對象被封裝在_connection -ptr接口、_commndptr接口和_recordsetptr接口這三個基本接口中。
3.2 ado連接數(shù)據(jù)庫
(1) 導(dǎo)入ado庫
使用ado前需要在stdafx.h頭文件中導(dǎo)入該庫,只需利用import指令將此動態(tài)鏈接庫導(dǎo)入,具體代碼如下:
#import“c:program filescommon filessystemadomsado15.dll”o_namespaceename(“eof”,“adoeof”)
(2) 初始化com庫
ado本身是一個com組件,在使用時需要初始化com庫,需要調(diào)用coinitialize函數(shù)來實現(xiàn)。
:: coinitialize(null);
(3) 創(chuàng)建connection對象并連接數(shù)據(jù)庫
創(chuàng)建connection對象:
_connectionptr m_pconnection;
m_pconnection.createinstanc -e(“adodb.connection”);
連接數(shù)據(jù)庫:
_pconnection-》open(“provide -r=microsoft.jet.oledb.4.0;datasource = picture.mdb”, “”,“”, admodeunknown);
(4) 執(zhí)行操作
m_precordset.createinstance(“adodb.recordset”);
m_precordset-》open((_variant_t)strsql,_variant_t((idispatch*)theapp.m_pconnection,true),adopendynamic,adlockoptimistic,adcmdtext);
3.3 圖像數(shù)據(jù)的存儲
對于一般數(shù)據(jù),可以通過recordset對象的getcollect()和putcollect()函數(shù)來對數(shù)據(jù)進(jìn)行存儲和讀取,而對于圖像數(shù)據(jù),就需要使用field對象中的appendchunk函數(shù)來進(jìn)行存儲和getchunk函數(shù)來進(jìn)行讀取。圖像數(shù)據(jù)的存儲過程可以分為以下三步:
(1) 打開圖像數(shù)據(jù),獲得數(shù)據(jù)長度;申請緩沖區(qū),把文件讀入此緩沖區(qū)。
(2) 構(gòu)造一個一維數(shù)組,把緩沖區(qū)中的數(shù)據(jù)拷貝到數(shù)組中,然后構(gòu)造一個變體對象,用數(shù)組來對此對象賦值。
(3) 調(diào)用appendchunk函數(shù),把數(shù)據(jù)存入recordset對象的本地緩沖區(qū)。
具體實現(xiàn)代碼如下:
if(m_pic.m_ipicture != null) m_pic.freepicturedata();
if(f.open(m_strphotopath, cfile::moderead |
cfile::typebinary, &e)) //打開了一個圖像文件
{nsize = f.getlength();//先得到圖像數(shù)據(jù)長度
byte * pbuffer = new byte [nsize];//根據(jù)數(shù)據(jù)的長度申請緩沖區(qū)
if (f.read(pbuffer, nsize) 》 0 )//把圖像數(shù)據(jù)讀到緩沖區(qū)
{byte *pbuf = pbuffer;//把pbuffer里的jpg數(shù)據(jù)放到庫中
variant varblob;
safearray *psa;
safearraybound rgsabound[1];//創(chuàng)建safearray對象
if(pbuf)
{rgsabound[0].llbound = 0;
rgsabound[0].celements = nsize;
psa = safearraycreate(vt_ui1, 1, rgsabound);
for (long i = 0; i 《 (long)nsize; i++)safearrayputelement (psa, &i, pbuf++);
varblob.vt = vt_array | vt_ui1;
varblob.parray = psa;
m_precordset-》getfields()-》getitem(“picture”)-》appendchunk(varblob);
m_precordset-》update();//更新數(shù)據(jù)庫
}
delete [] pbuffer; //釋放緩沖區(qū)
pbuf=0;
}
f.close();
}
3.4 圖像數(shù)據(jù)的讀取
圖像數(shù)據(jù)的讀取過程可以分為以下三歩:
(1) 獲得picture字段中圖像文件的實際長度。
(2) 調(diào)用getchunk函數(shù),返回變體對象。
(3) 獲得數(shù)組數(shù)據(jù)指針,把數(shù)據(jù)寫入打開的文件中。
具體實現(xiàn)代碼如下:
long nsize=m_precordset-》getfields()-》getitem(“picture ”)-》actualsize;//獲得圖像文件的長度
if(nsize1 》 0)
{ varblob=m_precordset-》getfields()-》getitem
(“picture ”)-》getchunk(nsize);//獲得varblob的值
if(varblob.vt == (vt_array | vt_ui1))
{char *pbuf = null;
lpvoid pbuf2 = null;
safearrayaccessdata(varblob.parray,
(void **)&pbuf);//獲得數(shù)組數(shù)據(jù)指針
hglobal=globalalloc(gmem_moveable,nsize);
pbuf2 = globallock(hglobal);
memcpy(pbuf2,pbuf,nsize1);//復(fù)制數(shù)據(jù)到緩沖區(qū)
::globalunlock(hglobal);
safearrayunaccessdata (varblob.parray);
hr=::createstreamonhglobal( hglobal,true, &pstream );
hr=::oleloadpicture( pstream, nsize1, true, iid_ipicture, (lpvoid * )&ppicture );
ppicture-》render(pdc-》m_hdc,0,0,rect.width(),rect.height(),0,nheight,nwidth,-nheight,null);//將圖像數(shù)據(jù)顯示在用戶界面
}
}
4 、應(yīng)用mfc odbc訪問圖像數(shù)據(jù)
4.1 mfc odbc的介紹
odbc(open database connect-ivity)是微軟公司開放服務(wù)結(jié)構(gòu)中有關(guān)數(shù)據(jù)庫的一個組成部分。mfc的odbc類對較復(fù)雜的odbcapi進(jìn)行了封裝,提供了簡化的調(diào)用接口,方便了數(shù)據(jù)庫應(yīng)用程序的開發(fā)。
mfcodbc類主要包括cdatabase類、crecordset類和cfieldexchange類。cdatabase類的主要功能是建立與數(shù)據(jù)庫的連接;crecordset類針對數(shù)據(jù)源中記錄集,它負(fù)責(zé)對記錄的操作;cfieldexchange類負(fù)責(zé)crecordset與數(shù)據(jù)源的數(shù)據(jù)交換。
4.2 配置odbc數(shù)據(jù)源
用odbc管理器定義了一個數(shù)據(jù)源,管理器根據(jù)數(shù)據(jù)源所在的位置,數(shù)據(jù)庫類型及odbc驅(qū)動器等信息,建立起odbc與具體數(shù)據(jù)庫之間的聯(lián)系。在windowsxp的系統(tǒng)中,配置odbc數(shù)據(jù)源的過程為:控制面板—性能和維護(hù)—管理工具—數(shù)據(jù)源(odbc)。
4.3 odbc連接數(shù)據(jù)庫
創(chuàng)建cdatabase對象并連接數(shù)據(jù)庫:
cdatabase m_db;//定義的數(shù)據(jù)庫全局變量
cbitmap bitmap;//定義的圖像類全局變量,存儲圖像數(shù)據(jù)
m_db.open(null, false, false, “odbc;driver={microsoft accessdriver (*.mdb)};
dbq= picture.mdb”);
4.4 圖像數(shù)據(jù)的存儲
首先做一些變量的定義:
m_ picture為picture字段對應(yīng)的長二進(jìn)制成員變量;
m_hdata用來存放picture字段的數(shù)據(jù);
m_dwdatalength為picture字段的實際長度。
圖像數(shù)據(jù)的存儲過程可以分為以下三步:
(1) 打開圖像數(shù)據(jù),獲得數(shù)據(jù)長度;
(2) 申請緩沖區(qū),把文件讀入此緩沖區(qū);
(3) 通過調(diào)用update()刷新數(shù)據(jù)庫記錄。
具體實現(xiàn)代碼如下:
static char based_code s***ilter[]=“bitmap files(*.bmp)|*.bmp||”;
cdbimages dbimages(m_db);
cfiledialog fd(true,null,null,0,s***ilter,this);
if (idok != fd.domodal()) return;
dbimages.open();
dbimages.addnew();
cfile fileimage;
cfilestatus filestatus;
fileimage.open(fd.getpathname(), cfile::moderead);
fileimage.getstatus(filestatus);//獲得打開圖像文件的狀態(tài)
dbimages.m_ picture.m_dwdatal -ength =filestatus.m_size;
hglobal hglobal=globalalloc(gptr,filestatus.m_size);//分配內(nèi)存
dbimages.m_ picture.m_hdata = globallock(hglobal);//鎖定內(nèi)存
fileimage.readhuge(dbimages.m_picture.m_hdata,filestatus.m_size);//向緩沖區(qū)讀取圖像數(shù)據(jù)
dbimages.setfielddirty(&dbimages.m_ picture);//修改標(biāo)志數(shù)據(jù)庫
dbimages.setfieldnull(&dbim -ages.m_ picture,false);
dbimages.update();//刷新數(shù)據(jù)庫的記錄
globalunlock(hglobal);
dbimages.close();
4.5 圖像數(shù)據(jù)的讀取
圖像數(shù)據(jù)的讀取過程可以分為以下三步:
(1) 定義臨時文件。
(2) 向臨時文件中存放從數(shù)據(jù)庫中讀取到的圖像數(shù)據(jù)。
(3) 取出圖像數(shù)據(jù),獲得圖像數(shù)據(jù)的寬度和高度。
具體實現(xiàn)代碼如下:
cdbimages dbimages(m_db);
cstring strfilename ;
i=1;
strfilename.format(“%s”,i)dbimages.open();
if (dbimages.iseof())afxmessagebox(“unable to get image from picture”);
else{char tmppath[_max_path 1];
gettemppath(_max_path,tmppath);
strfilename.insert(0,tmppath);
cfile outfile(strfilename,cfile::modecreate
|cfile::modewrite);
lpstr buffer=(lpstr)globallock(dbimages.m_ picture.m_hdata);
outfile.writehuge(buffer,dbima -ges.m_picture.m_dwdatalength);
globalunlock(dbimages.m_ picture.m_hdata);
outfile.close();
hbitmap hbm=(hbitmap)::loa -dimage(null,strfilename,image_bitmap, 0, 0, lr_loadfromfile);
if (bitmap.attach(hbm)) { bitmap bm;
bitmap.getbitmap(&bm); //獲得圖像的尺寸
bitmap.setbitmapdimension(bm.bmwidth,bm.bmheight);//獲得圖像的寬度和高度
}
5 、結(jié)束語
介紹了使用ado技術(shù)和mfcodbc在access中存儲和讀取圖像數(shù)據(jù)的方法以及部分程序代碼。這兩種訪問方法在vc++7.0的環(huán)境下進(jìn)行調(diào)試結(jié)果顯示,使用ado來訪問數(shù)據(jù)庫中的圖像數(shù)據(jù)更加的方便,更加的高效,使用mfcodbc方法訪問大對象數(shù)據(jù)的速度相對較慢。在開發(fā)擴展時會發(fā)現(xiàn)mfcodbc不能用于其他的非關(guān)系數(shù)據(jù)庫,使用的范圍相對較窄。這兩種方法還可以應(yīng)用于其他二進(jìn)制大對象數(shù)據(jù)的訪問。
責(zé)任編輯:gt
-
數(shù)據(jù)
+關(guān)注
關(guān)注
8文章
7067瀏覽量
89110 -
視頻
+關(guān)注
關(guān)注
6文章
1947瀏覽量
72939 -
存儲
+關(guān)注
關(guān)注
13文章
4320瀏覽量
85903
發(fā)布評論請先 登錄
相關(guān)推薦
評論