在VC進行數據庫編程中對圖像的處理一直是個難點,經常有朋友為如何向數據庫中添加圖像或從數據庫中讀取圖像數據並顯示處理等問題所困擾,目前關於VC數據庫編程的圖書不少,但很少有涉及圖象問題的,本文針對這一現狀,介紹了如何解決上述的問題,希望可以對朋友們有所幫助。
本文以ACESS97為例,采用ODBC方式打開數據庫Photo,數據庫Photo的images表中含有一個image的圖像字段,在剛打開時顯示第一條記錄。存取操作時關鍵是確定數據的長度,讀數據操作時圖像數據可以從記錄集的圖像字段對應的長二進制成員變量m_Image的成員m_hData得到,這個句柄變量存放分配給圖像字段的數據,該對象的另一個成員變量m_dwDataLength為該字段的實際長度,在某些情況下,m_dwDataLength的值有可能小於m_hData中的數據長度值。具體內容可以參考MSND。以下實現的部分關鍵代碼。首先請看 CdbImages記錄集的定義:
CdbImages::CdbImages(CDatabase* pdb)
: CRecordset(pdb)
{
//{{AFX_FIELD_INIT(CdbImages)
m_nFields = 2;//數據庫中有兩個字段
//}}AFX_FIELD_INIT
m_nDefaultType = snapshot;//數據庫以快照形式打開
}
CString CdbImages::GetDefaultConnect()
{
return _T("ODBC;DSN=MS Access Database");//以ODBC操作數據庫
}
CString CdbImages::GetDefaultSQL()
{
return _T("[Images]");//默認的操作是連接到數據庫中的Images表
}
void CdbImages::DoFieldExchange(CFieldExchange* pFX)
{
//{{AFX_FIELD_MAP(CdbImages)
pFX->SetFieldType(CFieldExchange::outputColumn);
RFX_LongBinary(pFX, _T("[Image]"), m_Image);//記錄集中的成員對象和表中的OLE字段采用長二進制交換。
//}}AFX_FIELD_MAP
}
///////////////////////////////////////////////////////////
//從數據庫中顯示圖像數據
CDatabase m_DB;//定義的數據庫全局變量
CBitmap Bitmap;//定義的圖像類全局變量,用以存儲圖像數據
void CImageView::OnOpenDateBase()
{
m_DB.Open(NULL, //以ODBC形式打開數據庫
FALSE,
FALSE,
"ODBC;DRIVER={
Microsof Access Driver (*.mdb)};DBQ=Photo.mdb");
CdbImages dbImages(m_DB);
CString strFileName ;// 用來存放臨時文件名,以該臨時文件存放讀取的數據庫的圖像數據。
i=1;//I為臨時文件號
StrFileName.Format("%s",i)
dbImages.Open();
if (dbImages.IsEOF())
AfxMessageBox("Unable to get image from db");
else
{
char tmpPath[_MAX_PATH+1];
GetTempPath(_MAX_PATH,tmpPath);//得到臨時文件的目錄;
strFileName.Insert(0,tmpPath);//生成臨時文件名
CFile outFile(strFileName,CFile::modeCreate|CFile::modeWrite);
//向臨時文件寫數據,數據庫中數據讀操作時關鍵是知道數據的長度,這可以從記錄集的圖像字段對應的長二進制成員變量m_Image的成員dwDataLength得到;
LPSTR buffer = (LPSTR)GlobalLock(dbImages.m_Image.m_hData);
outFile.WriteHuge(buffer,dbImages.m_Image.m_dwDataLength);
GlobalUnlock(dbImages.m_Image.m_hData);
outFile.Close();
//定義圖像句柄;
HBITMAP hbm = (HBITMAP)::LoadImage(NULL,
strFileName,
IMAGE_BITMAP,
0,
0,
LR_LOADFROMFILE);
if (Bitmap.Attach(hbm))//將圖像句柄和Cbitmap對象聯系起來
{
BITMAP bm;
Bitmap.GetBitmap(&bm);//獲取圖像尺寸
Bitmap.SetBitmapDimension(bm.bmWidth,bm.bmHeight);
return;
}
Invalidate();//重畫屏幕
}
/////////////////////////////////////////////
void CImageView::OnDraw(CDC* pDC)//顯示圖像
{
CDC memDC;
memDC.CreateCompatibleDC(pDC);
CBitmap* pImage = memDC.SelectObject(Bitmap);
CSize imageSize = Bitmap.GetBitmapDimension();
pDC->BitBlt(0,0,imageSize.cx,imageSize.cy,&memDC,0,0,SRCCOPY);
memDC.SelectObject(pImage);
}
//////////////////////////////////////////////////////
//以下是向數據中添加圖像記錄
void CImageView::OnSaveDateBase()
{//打開圖像文件
static char BASED_CODE szFilter[] = "Bitmap Files (*.bmp)|*.bmp||";
CdbImages dbImages(m_DB);
CFileDialog fd(TRUE,NULL,NULL,0,szFilter,this);
if (IDOK != fd.DoModal())
return;
dbImages.Open();//打開數據庫
dbImages.AddNew();//添加新的記錄
CFile fileImage;
CFileStatus fileStatus;
fileImage.Open(fd.GetPathName(), CFile::modeRead);
fileImage.GetStatus(fileStatus);//得到打開的圖像文件的狀態信息;
dbImages.m_Image.m_dwDataLength = fileStatus.m_size;
HGLOBAL hGlobal= GlobalAlloc(GPTR,fileStatus.m_size);//申請存放圖像數據的空間。
dbImages.m_Image.m_hData = GlobalLock(hGlobal);//將該空間付給m_hData成員;
//向緩沖區讀圖像數據
fileImage.ReadHuge(dbImages.m_Image.m_hData,fileStatus.m_size);//向m_Image讀圖像數據。
dbImages.SetFieldDirty(&dbImages.m_Image);//標志數據庫已經修改;
dbImages.SetFieldNull(&dbImages.m_Image,FALSE);//設置記錄為NULL;
dbImages.Update();//刷新數據庫的記錄
GlobalUnlock(hGlobal);
dbImages.Close();
}
上述代碼在VC6.0,windows98下編譯通過,運行正常,有興趣的朋友可以稍加修改,將文中所述的功能添加到自己的應用程序當中去。