C++應用ADO完成存取圖片的辦法。本站提示廣大學習愛好者:(C++應用ADO完成存取圖片的辦法)文章只能為提供參考,不一定能成為您想要的結果。以下是C++應用ADO完成存取圖片的辦法正文
普通在網上查到的材料中向Server2000存儲圖片代碼比擬多,從數據庫中讀取圖片並顯示也很多,然則把圖片從數據庫中二進制數據轉換為原圖片保留在當地,就很少有C++代碼了。本文就此成績一步一步地講一講授決的辦法:
1、應用數據庫前的預備
我們應用ADO,是用_ConnectionPtr,_RecordsetPtr來把持數據庫的。還有一個_CommandPtr,本法式沒有應用它。
為了應用ADO,須要導入ADO靜態鏈接庫。在工程的stdafx.h文件中,添加以下代碼:
//導入ADO #import "C:\Program Files\Common Files\System\ado\msado15.dll"\ rename_namespace("ADOCG")rename("EOF","EndOfFile") using namespace ADOCG;
這些代碼聲明,在這個工程中應用ADO但不應用ADO的名字空間,而且為了不常數抵觸,將常數EOF更名為adoEOF。
再有就是要建一個簡略的數據庫,名字叫TestImage,外面有一個表Images,這個表有三個字段,分離是ID,Name,ImageData。
2、銜接數據庫
銜接數據庫的代碼可以放入一個函數中,在想挪用的處所挪用。普通不推舉在CAPP類的Initalize()裡銜接數據庫,在加入法式時封閉數據庫銜接。應當是在應用時銜接,應用完立時封閉。項目中m_pConn是_ConnectionPtr類型的變量。
BOOL OpenConnection() { if(m_pConn == NULL) { m_pConn.CreateInstance("ADODB.Connection"); //創立_ConnectionPtr的一個實例 } try { if(adStateClosed == m_pConn->State) //假如已封閉 { m_pConn->Open("driver={SQL Server};Server=HP-CADD722B76A0;DATABASE=TestImage;UID=sa;PWD=sa","","",adModeUnknown); //因數據庫而異 return true; } } catch(_com_error e) { AfxMessageBox(_T("銜接數據庫掉敗!")); return false; } }
3、翻開數據集,把持數據庫
在應用_RecordSetPtr對象m_pRecord時,必需先創立這類對象的一個實例:
m_pRecord.CreateInstance( __uuidof(RecordSet) ); CString strSQL; //獲得表中最年夜的id,下一次拔出時就用id+1 strSQL.Format(_T("Select count(*) as num, Max(ID) as maxid from Images")); try { m_pRecord->Open(strSQL.AllocSysString(), m_pConn.GetInterfacePtr(), adOpenDynamic, adLockUnspecified, adCmdText); } catch (_com_error e) { AfxMessageBox(_T("讀取最年夜的id異常")); eturn; } //從RecordSet中獲得數據數量和以後數據庫中最年夜的ID。 int num = m_pRecord->GetCollect("num"); int maxid; if (num != 0) { maxid = m_pRecord->GetCollect("maxid"); } else { maxid = 0; } strSQL.Format(_T("Select * from Images where ID = %d"), maxid); //上面向數據庫中拔出圖片等。 //起首從數據庫中讀id最年夜的那條數據,重要目標是為了將RecordSet初始化 m_pRecord.CreateInstance(__uuidof(Recordset));
下面這句必定要留意,由於上一次把一些數據放入m_pRecord中,這一次再放的時刻,要從新創立一次,不然數據格局要末不婚配,要末保存有上一次的數據,定位艱苦。
m_pRecord->Open(strSQL.AllocSysString(), m_pConn.GetInterfacePtr(), adOpenDynamic, adLockOptimistic, adCmdText); //這是AddNew辦法請求的 CString imagepath = _T("F:/200713454/20090326.bmp"); CString imagename = imagepath.Right(12); try { m_pRecord->AddNew(); //為記載集添加新的一行,更新時就會把這條新記載放到數據庫中 } catch (_com_error e) { AfxMessageBox(_T("不克不及拔出一條新的記載")); return; } try { //應用putcollect拔出非圖象數據,應用SetImage2DB拔出圖象數據 m_pRecord->PutCollect("ID", _variant_t(maxid+1)); m_pRecord->PutCollect("Name", _variant_t(imagename)); SetImage2DB(imagepath); } catch (_com_error e) { AfxMessageBox(_T("拔出圖片有異常")); return; } m_pRecord->Update(); //應用終了,封閉m_pRecord,並設置為NULL,最初封閉數據庫銜接 m_pRecord->Close(); m_pRecord = NULL; CloseConnection();
4、讀取圖片並存儲到當地盤算機
要將數據庫中的二進制數據變成圖片,最簡略的辦法就是用GDI+。GDI+有一個類是Image,可以用stream來創立對象,還可以用Save辦法保留到當地,所以這個類很相符須要。
要應用GDI+,須要做些設置。起首在VS2005的項目屬性中,加上gdiplus.lib。
然後在stdafx.h中添加代碼
#include <GdiPlus.h> using namespace Gdiplus;
在CApp類添加兩個變量:
GdiplusStartupInput m_gdiplusstartUpInput; ULONG_PTR m_GdiplusToken;
在CApp的InitInstance函數中添加
GdiplusStartup(&m_GdiplusToken, &m_gdiplusstartUpInput, NULL);
在ExitInstance函數中添加
GdiplusShutdown(m_GdiplusToken);
以下是讀取圖片數據並保留到當地的代碼完成:
OpenConnection(); m_pRecord.CreateInstance(__uuidof(Recordset)); CString strSQL; strSQL.Format(_T("Select * from Images where ID = 1")); try { m_pRecord->Open(strSQL.AllocSysString(), m_pConn.GetInterfacePtr(), adOpenDynamic, adLockOptimistic, adCmdText); } catch (_com_error e) { AfxMessageBox(_T("讀取圖片信息異常")); return; } LPVOID Data; char* pbuf = NULL; long lDatasize = m_pRecord->GetFields()->GetItem("ImageData")->ActualSize; //數據庫中圖象數據長度 CString imagename = m_pRecord->GetCollect("Name").bstrVal; if (lDatasize > 0) { _variant_t varBLOB; varBLOB = m_pRecord->GetFields()->GetItem("ImageData")->GetChunk(lDatasize); Data = new char[lDatasize+1]; if (varBLOB.vt == (VT_ARRAY|VT_UI1)) { SafeArrayAccessData(varBLOB.parray, (void **)&pbuf); memcpy(Data, pbuf, lDatasize); SafeArrayUnaccessData(varBLOB.parray); } } IStream* pStm; LONGLONG cb = lDatasize; HGLOBAL hGlobal = GlobalAlloc(GMEM_MOVEABLE, cb); LPVOID pvData; if (hGlobal != NULL) { pvData = GlobalLock(hGlobal); memcpy(pvData, Data, cb); GlobalUnlock(hGlobal); CreateStreamOnHGlobal(hGlobal, TRUE, &pStm); } else { AfxMessageBox(_T("Error")); return; } CLSID encoderClsid; GetEncoderClsid(L"image/bmp",&encoderClsid); //肯定編碼格局是bmp格局 Image image(pStm, TRUE); CString imagepath; imagepath.Format(_T("F:/200713454/%s"), imagename); image.Save(imagepath, &encoderClsid, NULL); //把image中的數據依照bmp編碼格局存到當地 m_pRecord->Close(); m_pRecord = NULL; CloseConnection();
下面存儲和讀取數據的代碼頂用到了兩個函數,GetEncoderClsid和SetImage2DB。它們的完成以下:
這個函數和下面的存/取函數都是一個類的成員函數,而m_pConn和m_pRecord是這個類的成員變量,所以
void CDlgTest::SetImage2DB(CString path) { VARIANT varChunk; SAFEARRAY* psa; SAFEARRAYBOUND rgsabound[1]; CFile f(path.operator LPCTSTR(),CFile::modeRead); BYTE bval[ChunkSize+1]; long uIsRead=0; while (1) { uIsRead=f.Read(bval,ChunkSize); if (uIsRead==0) break; rgsabound[0].cElements=uIsRead; rgsabound[0].lLbound=0; psa=SafeArrayCreate(VT_UI1,1,rgsabound); for (long index=0;index<uIsRead;index++) { if (FAILED(SafeArrayPutElement(psa,&index,&bval[index]))) AfxMessageBox(_T("毛病。")); } varChunk.vt =VT_ARRAY|VT_UI1; varChunk.parray=psa; try { m_pRecord->Fields->GetItem("ImageData")->AppendChunk(varChunk); } catch (_com_error e) { AfxMessageBox(_T("毛病。")); } ::VariantClear(&varChunk); ::SafeArrayDestroyData(psa); if (uIsRead<ChunkSize)break; } f.Close(); } INT CDlgTest::GetEncoderClsid(const WCHAR* format, CLSID* pClsid) { UINT num = 0; // number of image encoders UINT size = 0; // size of the image encoder array in bytes ImageCodecInfo* pImageCodecInfo = NULL; GetImageEncodersSize(&num, &size); if(size == 0) return -1; // Failure pImageCodecInfo = (ImageCodecInfo*)(malloc(size)); if(pImageCodecInfo == NULL) return -1; // Failure GetImageEncoders(num, size, pImageCodecInfo); for(UINT j = 0; j < num; ++j) { if( wcscmp(pImageCodecInfo[j].MimeType, format) == 0 ) { *pClsid = pImageCodecInfo[j].Clsid; free(pImageCodecInfo); return j; // Success } } free(pImageCodecInfo); return -1; // Failure }
至此就完成了存儲圖片和從數據庫中把圖片下載到當地的功效。