在Visual C++ 6.0下显示JPEG、GIF等格式标准的图像的一种实现起来比较简便的方法

转载 2011年09月21日 16:59:43
 
摘要:本文讲述了在Visual C++ 6.0下显示JPEG、GIF等格式标准的图像的一种实现起来比较简便的方法,对实现过程作有详细的说明。

  关键字:图像、JPEG、GIF、Microsoft Visual C++ 6.0 
 
  一、 引言
  JPEG图像压缩标准随然是一种有损图像压缩标准,但由于人眼视觉的不敏感,经压缩后的画质基本没有发生变化,很快便以较高的压缩率得到了广泛的认可。GIF格式虽然仅支持256色但它对于颜色较少的图像有着很高的压缩率,甚至超过JPEG标准,也得到了广泛的认同。但作为众多程序员的一个重要的开发工具--Microsoft Visual C++ 6.0的MFC库却仅对没有经过任何压缩的BMP位图文件有着良好的支持,可以读取、显示、存储甚至在内存中创建一块内存位图。由于BMP格式的图像没有经过任何的压缩,不论是作为程序的外部文件,还是作为程序的内部资源都要占据大量的空间,尤其是后者会大大增加可执行文件的长度。可以看出,如果能用经过压缩、具有较好的压缩率的JPEG或GIF格式的图像来取代BMP文件在VC中的应用,无疑还是很有吸引力的。
  二、 设计思路
  虽然有一些操作、处理JPEG、GIF等其他格式图像的Active X控件,但总的来说使用起来并不太方便,笔者经过实验摸索,总结出了一种借助于COM接口的OLE方法来实现上述功能的一种简便方法,现介绍如下以飨广大读者:
  下面我们要使用IPicture 的COM接口,有必要对该图像接口做些了解:该接口主要管理图像对象及其属性,图像对象为 位图、图标和图元等提供一种与语言无关的抽象。和标准的字体对象一样,系统也提供了对图像对象的标准实现。其主要的接口是IPicture和IPictureDisp,后者是由IDispatch接口派生以便通过自动化对图像的属性进行访问。图像对象也支持外部接口IPropertyNotifySink,以便用户能在图像属性发生改变时作出决定。图像对象也支持IPersistStream接口,所以它能从一个IStream接口的实例对象保存、装载自己,而IStream接口也支持对流对象的数据读写。 
  我们可以用函数OleLoadPicture从包含有图像数据的流中装载图像。该函数简化了基于流的图像对象的创建过程,可以创建一个新的图像对象并且用流中的内容对它进行初始化。其函数原型为:
STDAPI OleLoadPicture( IStream * pStream, //指向包含有图像数据的流的指针
LONG lSize, //从流中读取的字节数
BOOL fRunmode, //图像属性对应的初值
REFIID riid, //涉及到的接口标识,描述要返回的接口指针的类型 
VOID ppvObj // 在rrid中用到的接口指针变量的地址) ; 
  三、 具体的实现
  在显示图像之前,首先要获取到图像文件的存放路径,这里采用标准的文件打开对话框来选取图像文件,文件名存放在CString 型的变量m_sPath中:
CFileDialog dlg(TRUE,"jpg","*.jpg",
OFN_HIDEREADONLY|OFN_OVERWRITEPROMPT,
"JPEG文件(*.jpg)|*.jpg|GIF文件(*.gif)|*.gif||",NULL);
if(dlg.DoModal()==IDOK)
{
 m_sPath=dlg.GetPathName();
 Invalidate();
} 
  为简单计,图形显示的代码直接在视类中的OnDraw中编写,首先打开文件并判断文件的可用性,并把文件内容放到流接口IStream的对象pStm中:
IStream *pStm; 
CFileStatus fstatus; 
CFile file; 
LONG cb; 
……
if (file.Open(m_Path,CFile::modeRead)&&file.GetStatus(m_Path,fstatus)&& ((cb = fstatus.m_size) != -1)) 
{
 
 HGLOBAL hGlobal = GlobalAlloc(GMEM_MOVEABLE, cb); 
 LPVOID pvData = NULL;
 
 if (hGlobal != NULL)
 
 {
 
  if ((pvData = GlobalLock(hGlobal)) != NULL)
 
  { 
   file.ReadHuge(pvData, cb); 
   GlobalUnlock(hGlobal); 
   CreateStreamOnHGlobal(hGlobal, TRUE, &pStm); 
  }
 }
} 

  然后,就直接调用OleLoadPicture函数从流中装载图像: 

IPicture *pPic; 
……
OleLoadPicture(pStm,fstatus.m_size,TRUE,IID_IPicture,(LPVOID*)&pPic)); 

  由于该函数有时会导致失败,所以应当用SUCCEEDED宏来做一些适当的保护工作,只有在数据装载成功的前提下才能继续下面的图像显示工作:

if(SUCCEEDED(OleLoadPicture(pStm,fstatus.m_size,TRUE,IID_IPicture,(LPVOID*)&pPic)))
{ 
 OLE_XSIZE_HIMETRIC hmWidth; 
 OLE_YSIZE_HIMETRIC hmHeight; 
 pPic->get_Width(&hmWidth); 
 pPic->get_Height(&hmHeight); 
 double fX,fY; 
 ……
 fX = (double)pDC->GetDeviceCaps(HORZRES)*(double)hmWidth/((double)pDC->GetDeviceCaps(HORZSIZE)*100.0); 
 fY = (double)pDC->GetDeviceCaps(VERTRES)*(double)hmHeight/((double)pDC->GetDeviceCaps(VERTSIZE)*100.0); 
 if(FAILED(pPic->Render(*pDC,0,0,(DWORD)fX,(DWORD)fY,0,hmHeight,hmWidth,-hmHeight,NULL))) {
  AfxMessageBox("渲染图像失败!"); 
  pPic->Release(); 
 } 
 else 
 AfxMessageBox("从流中装载图像失败!"); 
}  

  其中,显示工作主要是由IPicture接口对象的Render函数来完成的,该函数主要用来将图片的指定部分画到指定的设备环境的指定位置。原型如下: 

HRESULT Render( HDC hdc, //渲染图像用的设备环境句柄
long x, //在hdc上的水平坐标
long y, //在hdc上的垂直坐标
long cx, //图像宽度
long cy, //图像高度
OLE_XPOS_HIMETRIC xSrc, //在源图像上的水平偏移
OLE_YPOS_HIMETRIC ySrc, //在源图像上的垂直偏移
OLE_XSIZE_HIMETRIC cxSrc,//在源图像上水平拷贝的数量
OLE_YSIZE_HIMETRIC cySrc,//在源图像上垂直拷贝的数量
LPCRECT prcWBounds //指向目标图元设备环境句柄的指针); 

  小结:

  到此为止,通过上述代码已经能够在程序的客户区内显示JPEG、GIF等标准的图像了,但对于有多帧图片(即有动画)的GIF格式的图像,目前还只能显示第一帧,如要完整的显示GIF 动画的全过程,还需要外部Active X控件的支持。 

PS:本文引至于天极网。 还有给你推荐一个编程者的好去处,多去“CSDN”上混,呵呵。

还有对图片的EXIF信息读取看下楼

这个类中构造了两个结构MetadataDetail 和 Metadata  ,前者是为了存储EXIF中某元素信息的三种格式,起到中间转化的作用,一个是十六进制的索引信息,一个是没用处理过的信息代码,一个是显示信息。后面的那个结构就是存储一张图片所有EXIF信息元素的。 有两个方法LookupEXIFValue 和 GetEXIFMetaData ,前一个方法是处理特殊元素的对应显示的,后者从图片中读取相关信息然后填充到MetaData结构中。

using System;
using System.Drawing;
using System.Drawing.Imaging;
using System.Collections;
using System.ComponentModel;

namespace Test.Com
{
/// <summary>
/// 功能:获得图片EXIF信息
/// 作者:Rexsp
/// 创建日期:2004-03-20
/// </summary>
public class EXIFMetaData
{
  #region 构造函数
  /// <summary>
  /// 构造函数
  /// </summary>
  public EXIFMetaData()
  {
  }
  #endregion

  #region 数据转换结构
  /// <summary>
  /// 转换数据结构
  /// </summary>
  public struct MetadataDetail
  {
   public string Hex;//十六进制字符串
   public string RawValueAsString;//原始值串
   public string DisplayValue;//显示值串
  }
  #endregion

  #region EXIF元素结构
  /// <summary>
  /// 结构:存储EXIF元素信息
  /// </summary>
  public struct Metadata
  {
   public MetadataDetail EquipmentMake;
   public MetadataDetail CameraModel;
   public MetadataDetail ExposureTime;//曝光时间
   public MetadataDetail Fstop;
   public MetadataDetail DatePictureTaken;
   public MetadataDetail ShutterSpeed;// 快门速度
   public MetadataDetail MeteringMode;//曝光模式
   public MetadataDetail Flash;//闪光灯
   public MetadataDetail XResolution;
   public MetadataDetail YResolution;
   public MetadataDetail ImageWidth;//照片宽度
   public MetadataDetail ImageHeight;//照片高度

   public MetadataDetail FNumber;//  f值,光圈数
   public MetadataDetail ExposureProg;//  曝光程序
   public MetadataDetail SpectralSense;//  
   public MetadataDetail ISOSpeed;//  ISO感光度
   public MetadataDetail OECF;//  
   public MetadataDetail Ver;//  EXIF版本
   public MetadataDetail CompConfig;//  色彩设置
   public MetadataDetail CompBPP;//  压缩比率
   public MetadataDetail Aperture;//  光圈值
   public MetadataDetail Brightness;//  亮度值Ev
   public MetadataDetail ExposureBias;//  曝光补偿
   public MetadataDetail MaxAperture;//  最大光圈值

   public MetadataDetail SubjectDist;// 主体距离
   public MetadataDetail LightSource;//  白平衡
   public MetadataDetail FocalLength;//  焦距
   public MetadataDetail FPXVer;//  FlashPix版本
   public MetadataDetail ColorSpace;//  色彩空间
   public MetadataDetail Interop;//  
   public MetadataDetail FlashEnergy;//  
   public MetadataDetail SpatialFR;//  
   public MetadataDetail FocalXRes;//  
   public MetadataDetail FocalYRes;//  
   public MetadataDetail FocalResUnit;//  
   public MetadataDetail ExposureIndex;//  曝光指数
   public MetadataDetail SensingMethod;//  感应方式
   public MetadataDetail SceneType;//  
   public MetadataDetail CfaPattern;//  
  }
  #endregion

  #region 查找EXIF元素值
  public string LookupEXIFValue(string Description, string Value)
  {
   string DescriptionValue = null;

   switch(Description)
   {
    case "MeteringMode":

     #region  MeteringMode
    {
     switch(Value)
     {
      case "0":
       DescriptionValue = "Unknown";break;
      case "1":
       DescriptionValue = "Average";break;
      case "2":
       DescriptionValue = "Center Weighted Average";break;
      case "3":
       DescriptionValue = "Spot";break;
      case "4":
       DescriptionValue = "Multi-spot";break;
      case "5":
       DescriptionValue = "Multi-segment";break;
      case "6":
       DescriptionValue = "Partial";break;
      case "255":
       DescriptionValue = "Other";break;
     }
    }
     #endregion
     
     break;
    case "ResolutionUnit":

     #region ResolutionUnit
    {
     switch(Value)
     {
      case "1":
       DescriptionValue = "No Units";break;
      case "2":
       DescriptionValue = "Inch";break;
      case "3":
       DescriptionValue = "Centimeter";break;
     }
    }

     #endregion

     break;
    //省略N行相似代码

   }
   return DescriptionValue;
  }
  #endregion

  #region 取得图片的EXIF信息
  public Metadata GetEXIFMetaData(string PhotoName)
  {
   // 创建一个图片的实例
   System.Drawing.Image MyImage = System.Drawing.Image.FromFile(PhotoName);
   // 创建一个整型数组来存储图像中属性数组的ID
   int[] MyPropertyIdList = MyImage.PropertyIdList;
   //创建一个封闭图像属性数组的实例
   PropertyItem[] MyPropertyItemList = new PropertyItem[MyPropertyIdList.Length];
   //创建一个图像EXIT信息的实例结构对象,并且赋初值

   #region 创建一个图像EXIT信息的实例结构对象,并且赋初值
   Metadata MyMetadata = new Metadata();
   MyMetadata.EquipmentMake.Hex = "10f";
   MyMetadata.CameraModel.Hex = "110";
   MyMetadata.DatePictureTaken.Hex = "9003";
   MyMetadata.ExposureTime.Hex = "829a";
   MyMetadata.Fstop.Hex = "829d";
   MyMetadata.ShutterSpeed.Hex = "9201";
   MyMetadata.MeteringMode.Hex = "9207";
   MyMetadata.Flash.Hex = "9209";
   MyMetadata.FNumber.Hex = "829d"; // 
   MyMetadata.ExposureProg.Hex = ""; // 
   MyMetadata.SpectralSense.Hex = "8824"; // 
   MyMetadata.ISOSpeed.Hex = "8827"; // 
   MyMetadata.OECF.Hex = "8828"; // 
   MyMetadata.Ver.Hex = "9000"; // 
   MyMetadata.CompConfig.Hex = "9101"; // 
   MyMetadata.CompBPP.Hex = "9102"; // 
   MyMetadata.Aperture.Hex = "9202"; // 
   MyMetadata.Brightness.Hex = "9203"; // 
   MyMetadata.ExposureBias.Hex = "9204"; // 
   MyMetadata.MaxAperture.Hex = "9205"; // 
   MyMetadata.SubjectDist.Hex = "9206"; // 
   MyMetadata.LightSource.Hex = "9208"; // 
   MyMetadata.FocalLength.Hex = "920a"; // 
   MyMetadata.FPXVer.Hex = "a000"; // 
   MyMetadata.ColorSpace.Hex = "a001"; // 
   MyMetadata.FocalXRes.Hex = "a20e"; // 
   MyMetadata.FocalYRes.Hex = "a20f"; // 
   MyMetadata.FocalResUnit.Hex = "a210"; // 
   MyMetadata.ExposureIndex.Hex = "a215"; // 
   MyMetadata.SensingMethod.Hex = "a217"; // 
   MyMetadata.SceneType.Hex = "a301";
   MyMetadata.CfaPattern.Hex = "a302";
   #endregion
   // ASCII编码
   System.Text.ASCIIEncoding Value = new System.Text.ASCIIEncoding();
            
   int index = 0;
   int MyPropertyIdListCount=MyPropertyIdList.Length;
   if(MyPropertyIdListCount!=0)
   {
    foreach (int MyPropertyId in MyPropertyIdList)
    {
     string hexVal = "";
     MyPropertyItemList[index] = MyImage.GetPropertyItem(MyPropertyId);
     #region 初始化各属性值
     string myPropertyIdString=MyImage.GetPropertyItem(MyPropertyId).Id.ToString("x");
     switch(myPropertyIdString)
     {
      case "10f":
      {
       MyMetadata.EquipmentMake.RawValueAsString =BitConverter.ToString(MyImage.GetPropertyItem (MyPropertyId).Value);
       MyMetadata.EquipmentMake.DisplayValue = Value.GetString(MyPropertyItemList[index].Value);
       break;
      }

      case "110":
      {
       MyMetadata.CameraModel.RawValueAsString =BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
       MyMetadata.CameraModel.DisplayValue =Value.GetString(MyPropertyItemList[index].Value);
       break;
      }

      case "9003":
      {
       MyMetadata.DatePictureTaken.RawValueAsString =BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
       MyMetadata.DatePictureTaken.DisplayValue =Value.GetString(MyPropertyItemList[index].Value);
       break;
      }
    //省略N行相似代码
     }
     #endregion
     
     index++;
    }
   }

   MyMetadata.XResolution.DisplayValue = MyImage.HorizontalResolution.ToString();
   MyMetadata.YResolution.DisplayValue = MyImage.VerticalResolution.ToString();
   MyMetadata.ImageHeight.DisplayValue = MyImage.Height.ToString();
   MyMetadata.ImageWidth.DisplayValue = MyImage.Width.ToString();
   MyImage.Dispose();
   return MyMetadata;
  }
  #endregion
}
}

然后就是个调用的问题,有了这个类,我如何读取图片的EXIF信息呢?代码如下:
   EXIFMetaData em = new EXIFMetaData();
   string filePath=Server.MapPath("Test.jpg");//这里可以动态传递图片路径的
   EXIFMetaData.Metadata m = em.GetEXIFMetaData(filePath);//这里就是调用,传图片绝对路径
   string exif = m.Ver.DisplayValue;
   string camera = m.CameraModel.DisplayValue;
   string model = m.CameraModel.DisplayValue;
   string aperture = m.Aperture.DisplayValue;
   string shutter = m.ShutterSpeed.DisplayValue;
   string sensitive = m.ExposureIndex.DisplayValue;

此代码经过测试,功能正常。

Visual C++中实现对图像数据的读取显示

作者:jiaohe2000 转自:http://blog.csdn.net/jiaohe2000/article/details/1733517 在利用VC进行数据库编程时,经常需要处理数...
  • whw8007
  • whw8007
  • 2014年01月12日 19:03
  • 716

自己用C++实现BaseLine Jpeg解码(要点总结)

忙活这几天,终于自己用C++完整实现了BaseLine的Jpeg解码算法,中间阅读了一些论文和网上的帖子,发现其中有很多没有说清楚的地方,自己在实现的过程中也受到网上资料的很多误导,现在自己完整地把J...
  • xiaohaowudi
  • xiaohaowudi
  • 2014年02月11日 00:04
  • 1323

jpg、jpeg、png、gif、bmp、tiff、ai、cdr、eps 图片格式的区别

jpg、jpeg、png、gif、bmp、tiff、ai、cdr、eps这些图片格式是最常用的,也是最常见的,几乎每一天都要与他们打交道。 刚刚入门的新人通常不知道在什么地方如何使用他们或者说如何更有...
  • zmx729618
  • zmx729618
  • 2017年02月28日 10:59
  • 2024

BMP、JPEG、PNG、GIF格式总结

一、BMP格式(Bitmap) BMP是Windows系统的标准图像文件格式,因此,BMP格式与现有Windows程序广泛兼容。BMP图像采用位映射存储格式,图像深度有1bit、4Bit、8Bit、...
  • u011974126
  • u011974126
  • 2015年08月07日 17:07
  • 1181

JPEG图像压缩算法流程详解

转自http://blog.csdn.net/carson2005/article/details/7753499 JPEG是Joint Photographic Exports Group的英文缩...
  • qingkongyeyue
  • qingkongyeyue
  • 2017年04月22日 15:23
  • 4582

VC下加载JPG/GIF/PNG图片的两种方法(转)

本文转自 http://blog.sina.com.cn/s/blog_6582aa410100huil.html。 仅管VC有提供相应的API和类来操作bmp位图、图标和(增强)元...
  • zhongbin104
  • zhongbin104
  • 2013年03月28日 13:09
  • 3035

【原创】Jpeg渐进式图像数据解析

为了更好的优化客户端体验,客户端在图像压缩的时候采用了渐进式Jpeg压缩。渐进式Jpeg的好处是,只需要很少的一部分数据包,就能够解码出一副完整的图像,随着数据的增加,图像会不断变清晰。渐进式图像还有...
  • APIX_CN
  • APIX_CN
  • 2015年11月11日 13:25
  • 734

JPEG图像压缩和解压缩操作

PEG图像的解压缩操作   解压缩操作过程 1.        为JPEG对象分配空间并初始化 2.        指定解压缩数据源 3.        获取文件信息 4.        ...
  • u012372584
  • u012372584
  • 2016年02月01日 15:11
  • 4067

jpeg图像压缩原理

step1.颜色模式转换 JPEG只支持YUV颜色模式(准确说是YCbCr颜色模式)的数据结构,而不支持RGB图像数据结构,所以在将彩色图像进行压缩之前,必须先对颜色模式进行数据转换。YUV色彩模型来...
  • Quason
  • Quason
  • 2015年06月24日 09:49
  • 4917

VC下显示JPG,GIF图象的一种简便方法

作者:jiaohe2000 转自:http://blog.csdn.net/jiaohe2000/article/details/1884756 一、 引言    JPEG图像压...
  • whw8007
  • whw8007
  • 2014年01月12日 18:45
  • 407
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:在Visual C++ 6.0下显示JPEG、GIF等格式标准的图像的一种实现起来比较简便的方法
举报原因:
原因补充:

(最多只允许输入30个字)