Visual C 中实现对图像数据的读取显示
创始人
2024-07-31 03:11:43
0

在利用VC进行数据库编程时,经常需要处理数据库中的图像数据,将该图像从数据库中读取出来并显示,图像数据与文本字段不同,它是作为OLE字段在数据库中存储,通过数据集对象的成员变量自动交换得到的图像数据,得到的数据并不能直接显示,如何处理图像数据,一直是数据库编程中的一个难点。

目前关于VC进行数据库编程的资料不少,但很少涉及图像数据的操作,笔者针对一现状,结合自己开发的一个项目,解决了如何显示数据库中的图像这一问题,本文以操作ACESS数据库为例子,讲解一下自己的实现思路,希望对爱好VC编程的朋友们有所帮助,以起到抛砖引玉的作用。

为了简化问题,该数据库的表中只有一个名为Images的OLE字段,我使用DAO连接操作数据库,读取的图像数据显示在一个对话框上,至于使用ODBC、DAO还是ADO,这要根据具体情况而定,但无论使用哪一种,对图像的显示来说,实现的过程是大同小异的。

由于篇幅有限,文章中对如何实现数据库的连接不再作具体的说明,有兴趣的读者朋友可以参考VC数据库编程的资料。实现过程中,首先定义一个CDaoRecordset的子类 CimageData如下:

  1. class CimageData : public CDaoRecordset  
  2. {  
  3. public:  
  4. CimageData (CDaoDatabase* pDatabase = NULL);  
  5. DECLARE_DYNAMIC(CimageData)  
  6. file://{{AFX_FIELD(CimageData, CDaoRecordset)  
  7. CByteArray m_Images;//声明字节数组用来存放图像数据  
  8. file://}}AFX_FIELD  
  9. // Overrides  
  10. // ClassWizard generated virtual function overrides  
  11. file://{{AFX_VIRTUAL(CimageData)  
  12. public:  
  13. virtual CString GetDefaultDBName();  
  14. virtual CString GetDefaultSQL();  
  15. virtual void DoFieldExchange(CDaoFieldExchange* pFX);  
  16. file://}}AFX_VIRTUAL 

该类的实现为:

  1. CimageData:: CimageData (CDaoDatabase* pdb)  
  2. : CDaoRecordset(pdb)  
  3. {  
  4. file://{{AFX_FIELD_INIT(CimageData)  
  5. m_nFields = 1;//数据库的表中仅有一个字段  
  6. file://}}AFX_FIELD_INIT  
  7. m_nDefaultType = dbOpenDynaset;//以动态集方式打开数据库  
  8. }  
  9. CString CimageData::GetDefaultDBName()  
  10. {  
  11. return _T("E:\\IMAGES.mdb");//默认的ACESS数据库在E盘,名为IMAGES  
  12. }  
  13. CString CimageData::GetDefaultSQL()  
  14. {  
  15. return _T("[Table]");//默认打开数据库中名为"Table"的表  
  16. }  
  17. void CimageData::DoFieldExchange(CDaoFieldExchange* pFX)  
  18. {  
  19. file://{{AFX_FIELD_MAP(CimageData)  
  20. pFX->SetFieldType(CDaoFieldExchange::outputColumn);  
  21. DFX_Binary(pFX, _T("[Images]"), m_Images);//以二进制方式在Images字段和m_Images变量间交换数据  
  22. file://}}AFX_FIELD_MAP  

 

有了该类,就可以定义相应的对象来与数据库中的图像字段交换数据,下面定义的函数GetImageData()说明了如何根据读取的OLE字段数据生成待显示的图像,需要注意的是该函数中使用的CBitmap类的变量Bitmap是预定义的一个全局变量:

  1. BOOL CImageDlg:: GetImageData(CByteArray & DBArray)  
  2. {  
  3. CByteArray Array;  
  4. Array.Copy( DBArray);  
  5. int HeaderLen = 78 + sizeof(BITMAPFILEHEADER); file://确定图像头信息的起始位置  
  6. Array.RemoveAt( 0, HeaderLen ); // 移动到图像头信息的起始位置  
  7. BITMAPINFOHEADER &bmiHeader = *(LPBITMAPINFOHEADER)Array.GetData() ;  
  8. BITMAPINFO &bmInfo = *(LPBITMAPINFO)Array.GetData() ;  
  9. file://得到图像数据的头信息  
  10. int nColors=bmiHeader.biClrUsed ? bmiHeader.biClrUsed : 1 《 bmiHeader.biBitCount;  
  11. file://确定图像的颜色数  
  12. LPVOID lpDIBBits;  
  13. if( bmInfo.bmiHeader.biBitCount > 8 )  
  14. lpDIBBits=(LPVOID)((LPDWORD)(bmInfo.bmiColors+bmInfo.bmiHeader.biClrUsed)+  
  15. ((bmInfo.bmiHeader.biCompression == BI_BITFIELDS) ? 3 : 0));  
  16. else 
  17. lpDIBBits = (LPVOID)(bmInfo.bmiColors + nColors);  
  18. file://得到图像各个像素的具体数据  
  19. CClientDC dc(NULL);  
  20. HBITMAP hBmp = CreateDIBitmap( dc.m_hDC,  
  21. &bmiHeader,  
  22. CBM_INIT,  
  23. lpDIBBits,  
  24. &bmInfo,  
  25. DIB_RGB_COLORS);  
  26. file://生成位图句柄  
  27. Bitmap.Attach( hBmp );//将该句柄与定义的Bitmap对象联系在一起  
  28. Array.RemoveAll(); file://释放内存  
  29. return TRUE;  

有了上面的准备工作,现在可以实现图像的显示函数了,其实现如下:

  1. void CImageDlg::OnShowImage()  
  2. {  
  3. CimageData db;//定义记录集对象  
  4. db.Open();打开数据库  
  5. GetImageData(db.m_Images);//根据记录集对象的成员变量生成图像对象  
  6. file://以下是在对话框的固定区域显示图像  
  7. CPaintDC dc(this);  
  8. if (!(Bitmap.m_hObject == NULL))  
  9. { CDC dcMem;  
  10. dcMem.CreateCompatibleDC( &dc ); file://create a Memory Image  
  11. CBitmap* pbmpOld ;  
  12. BITMAP BmpSize ;  
  13. Bitmap.GetBitmap(&BmpSize); file://get Image Size  
  14. pbmpOld = dcMem.SelectObject(&Bitmap);  
  15. dc.StretchBlt( 20, 20, 200, 200, &dcMem, 0, 0, BmpSize.bmWidth, BmpSize.bmHeight, SRCCOPY);  
  16. dcMem.SelectObject( pbmpOld );  

以上代码中使用的数据库为ACESS97,程序在windows98、Visual C++6.0环境下编译通过,运行正常。

【编辑推荐】

  1. VC++获得当前系统时间的几种方案
  2. MVC+jQuery开发B/S系统:表单提交
  3. 浅谈怎样加快C++代码的编译速度
  4. C/C++是程序员必须掌握的语言吗?
  5. 再驳Linus:思科工程师对C++不得不说的事

相关内容

热门资讯

如何允许远程连接到MySQL数... [[277004]]【51CTO.com快译】默认情况下,MySQL服务器仅侦听来自localhos...
如何利用交换机和端口设置来管理... 在网络管理中,总是有些人让管理员头疼。下面我们就将介绍一下一个网管员利用交换机以及端口设置等来进行D...
施耐德电气数据中心整体解决方案... 近日,全球能效管理专家施耐德电气正式启动大型体验活动“能效中国行——2012卡车巡展”,作为该活动的...
Windows恶意软件20年“... 在Windows的早期年代,病毒游走于系统之间,偶尔删除文件(但被删除的文件几乎都是可恢复的),并弹...
20个非常棒的扁平设计免费资源 Apple设备的平面图标PSD免费平板UI 平板UI套件24平图标Freen平板UI套件PSD径向平...
德国电信门户网站可实时显示全球... 德国电信周三推出一个门户网站,直观地实时提供其安装在全球各地的传感器网络检测到的网络攻击状况。该网站...
着眼MAC地址,解救无法享受D... 在安装了DHCP服务器的局域网环境中,每一台工作站在上网之前,都要先从DHCP服务器那里享受到地址动...
为啥国人偏爱 Mybatis,... 关于 SQL 和 ORM 的争论,永远都不会终止,我也一直在思考这个问题。昨天又跟群里的小伙伴进行...