在Visual C++ 6.0下显示JPEG、GIF等格式标准的图像的一种实现起来比较简便的方法
2011-09-21 16:59
676 查看
摘要:本文讲述了在VisualC++6.0下显示JPEG、GIF等格式标准的图像的一种实现起来比较简便的方法,对实现过程作有详细的说明。 关键字:图像、JPEG、GIF、MicrosoftVisualC++6.0
一、引言
JPEG图像压缩标准随然是一种有损图像压缩标准,但由于人眼视觉的不敏感,经压缩后的画质基本没有发生变化,很快便以较高的压缩率得到了广泛的认可。GIF格式虽然仅支持256色但它对于颜色较少的图像有着很高的压缩率,甚至超过JPEG标准,也得到了广泛的认同。但作为众多程序员的一个重要的开发工具--MicrosoftVisualC++6.0的MFC库却仅对没有经过任何压缩的BMP位图文件有着良好的支持,可以读取、显示、存储甚至在内存中创建一块内存位图。由于BMP格式的图像没有经过任何的压缩,不论是作为程序的外部文件,还是作为程序的内部资源都要占据大量的空间,尤其是后者会大大增加可执行文件的长度。可以看出,如果能用经过压缩、具有较好的压缩率的JPEG或GIF格式的图像来取代BMP文件在VC中的应用,无疑还是很有吸引力的。
二、设计思路
虽然有一些操作、处理JPEG、GIF等其他格式图像的ActiveX控件,但总的来说使用起来并不太方便,笔者经过实验摸索,总结出了一种借助于COM接口的OLE方法来实现上述功能的一种简便方法,现介绍如下以飨广大读者:
下面我们要使用IPicture的COM接口,有必要对该图像接口做些了解:该接口主要管理图像对象及其属性,图像对象为位图、图标和图元等提供一种与语言无关的抽象。和标准的字体对象一样,系统也提供了对图像对象的标准实现。其主要的接口是IPicture和IPictureDisp,后者是由IDispatch接口派生以便通过自动化对图像的属性进行访问。图像对象也支持外部接口IPropertyNotifySink,以便用户能在图像属性发生改变时作出决定。图像对象也支持IPersistStream接口,所以它能从一个IStream接口的实例对象保存、装载自己,而IStream接口也支持对流对象的数据读写。
我们可以用函数OleLoadPicture从包含有图像数据的流中装载图像。该函数简化了基于流的图像对象的创建过程,可以创建一个新的图像对象并且用流中的内容对它进行初始化。其函数原型为:
STDAPIOleLoadPicture(IStream*pStream,//指向包含有图像数据的流的指针
LONGlSize,//从流中读取的字节数
BOOLfRunmode,//图像属性对应的初值
REFIIDriid,//涉及到的接口标识,描述要返回的接口指针的类型
VOIDppvObj//在rrid中用到的接口指针变量的地址);
三、具体的实现
在显示图像之前,首先要获取到图像文件的存放路径,这里采用标准的文件打开对话框来选取图像文件,文件名存放在CString型的变量m_sPath中:
CFileDialogdlg(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;
CFileStatusfstatus;
CFilefile;
LONGcb;
……
if(file.Open(m_Path,CFile::modeRead)&&file.GetStatus(m_Path,fstatus)&&((cb=fstatus.m_size)!=-1))
{
HGLOBALhGlobal=GlobalAlloc(GMEM_MOVEABLE,cb);
LPVOIDpvData=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_HIMETRIChmWidth;
OLE_YSIZE_HIMETRIChmHeight;
pPic->get_Width(&hmWidth);
pPic->get_Height(&hmHeight);
doublefX,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函数来完成的,该函数主要用来将图片的指定部分画到指定的设备环境的指定位置。原型如下:
HRESULTRender(HDChdc,//渲染图像用的设备环境句柄
longx,//在hdc上的水平坐标
longy,//在hdc上的垂直坐标
longcx,//图像宽度
longcy,//图像高度
OLE_XPOS_HIMETRICxSrc,//在源图像上的水平偏移
OLE_YPOS_HIMETRICySrc,//在源图像上的垂直偏移
OLE_XSIZE_HIMETRICcxSrc,//在源图像上水平拷贝的数量
OLE_YSIZE_HIMETRICcySrc,//在源图像上垂直拷贝的数量
LPCRECTprcWBounds//指向目标图元设备环境句柄的指针);
小结:
到此为止,通过上述代码已经能够在程序的客户区内显示JPEG、GIF等标准的图像了,但对于有多帧图片(即有动画)的GIF格式的图像,目前还只能显示第一帧,如要完整的显示GIF动画的全过程,还需要外部ActiveX控件的支持。
PS:本文引至于天极网。还有给你推荐一个编程者的好去处,多去“CSDN”上混,呵呵。
还有对图片的EXIF信息读取看下楼
这个类中构造了两个结构MetadataDetail和Metadata,前者是为了存储EXIF中某元素信息的三种格式,起到中间转化的作用,一个是十六进制的索引信息,一个是没用处理过的信息代码,一个是显示信息。后面的那个结构就是存储一张图片所有EXIF信息元素的。有两个方法LookupEXIFValue和GetEXIFMetaData,前一个方法是处理特殊元素的对应显示的,后者从图片中读取相关信息然后填充到MetaData结构中。
usingSystem;
usingSystem.Drawing;
usingSystem.Drawing.Imaging;
usingSystem.Collections;
usingSystem.ComponentModel;
namespaceTest.Com
{
///<summary>
///功能:获得图片EXIF信息
///作者:Rexsp
///创建日期:2004-03-20
///</summary>
publicclassEXIFMetaData
{
#region构造函数
///<summary>
///构造函数
///</summary>
publicEXIFMetaData()
{
}
#endregion
#region数据转换结构
///<summary>
///转换数据结构
///</summary>
publicstructMetadataDetail
{
publicstringHex;//十六进制字符串
publicstringRawValueAsString;//原始值串
publicstringDisplayValue;//显示值串
}
#endregion
#regionEXIF元素结构
///<summary>
///结构:存储EXIF元素信息
///</summary>
publicstructMetadata
{
publicMetadataDetailEquipmentMake;
publicMetadataDetailCameraModel;
publicMetadataDetailExposureTime;//曝光时间
publicMetadataDetailFstop;
publicMetadataDetailDatePictureTaken;
publicMetadataDetailShutterSpeed;//快门速度
publicMetadataDetailMeteringMode;//曝光模式
publicMetadataDetailFlash;//闪光灯
publicMetadataDetailXResolution;
publicMetadataDetailYResolution;
publicMetadataDetailImageWidth;//照片宽度
publicMetadataDetailImageHeight;//照片高度
publicMetadataDetailFNumber;//f值,光圈数
publicMetadataDetailExposureProg;//曝光程序
publicMetadataDetailSpectralSense;//
publicMetadataDetailISOSpeed;//ISO感光度
publicMetadataDetailOECF;//
publicMetadataDetailVer;//EXIF版本
publicMetadataDetailCompConfig;//色彩设置
publicMetadataDetailCompBPP;//压缩比率
publicMetadataDetailAperture;//光圈值
publicMetadataDetailBrightness;//亮度值Ev
publicMetadataDetailExposureBias;//曝光补偿
publicMetadataDetailMaxAperture;//最大光圈值
publicMetadataDetailSubjectDist;//主体距离
publicMetadataDetailLightSource;//白平衡
publicMetadataDetailFocalLength;//焦距
publicMetadataDetailFPXVer;//FlashPix版本
publicMetadataDetailColorSpace;//色彩空间
publicMetadataDetailInterop;//
publicMetadataDetailFlashEnergy;//
publicMetadataDetailSpatialFR;//
publicMetadataDetailFocalXRes;//
publicMetadataDetailFocalYRes;//
publicMetadataDetailFocalResUnit;//
publicMetadataDetailExposureIndex;//曝光指数
publicMetadataDetailSensingMethod;//感应方式
publicMetadataDetailSceneType;//
publicMetadataDetailCfaPattern;//
}#endregion
#region查找EXIF元素值
publicstringLookupEXIFValue(stringDescription,stringValue)
{
stringDescriptionValue=null;
switch(Description)
{
case"MeteringMode":
#regionMeteringMode
{
switch(Value)
{
case"0":
DescriptionValue="Unknown";break;
case"1":
DescriptionValue="Average";break;
case"2":
DescriptionValue="CenterWeightedAverage";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":
#regionResolutionUnit
{
switch(Value)
{
case"1":
DescriptionValue="NoUnits";break;
case"2":
DescriptionValue="Inch";break;
case"3":
DescriptionValue="Centimeter";break;
}
}
#endregion
break;
//省略N行相似代码
}
returnDescriptionValue;
}
#endregion
#region取得图片的EXIF信息
publicMetadataGetEXIFMetaData(stringPhotoName)
{
//创建一个图片的实例
System.Drawing.ImageMyImage=System.Drawing.Image.FromFile(PhotoName);
//创建一个整型数组来存储图像中属性数组的ID
int[]MyPropertyIdList=MyImage.PropertyIdList;
//创建一个封闭图像属性数组的实例
PropertyItem[]MyPropertyItemList=newPropertyItem[MyPropertyIdList.Length];
//创建一个图像EXIT信息的实例结构对象,并且赋初值
#region创建一个图像EXIT信息的实例结构对象,并且赋初值
MetadataMyMetadata=newMetadata();
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.ASCIIEncodingValue=newSystem.Text.ASCIIEncoding();
intindex=0;
intMyPropertyIdListCount=MyPropertyIdList.Length;
if(MyPropertyIdListCount!=0)
{
foreach(intMyPropertyIdinMyPropertyIdList)
{
stringhexVal="";
MyPropertyItemList[index]=MyImage.GetPropertyItem(MyPropertyId);
#region初始化各属性值
stringmyPropertyIdString=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();
returnMyMetadata;
}
#endregion
}
}
然后就是个调用的问题,有了这个类,我如何读取图片的EXIF信息呢?代码如下:
EXIFMetaDataem=newEXIFMetaData();
stringfilePath=Server.MapPath("Test.jpg");//这里可以动态传递图片路径的
EXIFMetaData.Metadatam=em.GetEXIFMetaData(filePath);//这里就是调用,传图片绝对路径
stringexif=m.Ver.DisplayValue;
stringcamera=m.CameraModel.DisplayValue;
stringmodel=m.CameraModel.DisplayValue;
stringaperture=m.Aperture.DisplayValue;
stringshutter=m.ShutterSpeed.DisplayValue;
stringsensitive=m.ExposureIndex.DisplayValue;
此代码经过测试,功能正常。
参考资料:
相关文章推荐
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 在VC下显示JPEG、GIF格式图像的一种简便方法
- 转贴:VC下显示JPEG、GIF图像的简便方法
- VC下显示JPEG、GIF图像的简便方法
- VC下显示JPG,GIF图象的一种简便方法
- VC下显示JPG,GIF图象的一种简便方法
- VC下显示JPG,GIF图象的一种简便方法
- wince下显示GIF、JPEG以及其它格式的图像文件
- VC 如何显示JPEG、GIF格式图像
- wince下显示GIF、JPEG以及其它格式的图像文件
- 图像格式jpg、jpeg、jpe、gif、png、png等有何不同?ps中那种图片格式可以保留图层?
- 解决ScrollView嵌套RecyclerView只显示一行的一种比较简单的方法
- Wuqi.Webdiyer.AspNetPager 比较标准的分页显示格式和属性设置
- Visual C++中实现对图像数据的读取显示
- 演示如何实现Matplotlib绘图并保存图像但不显示图形的方法