如何在vc中实现jpeg文件的显示和exif的读取

如何在vc中实现jpeg文件的显示和exif的读取,第1张

在显示图像之前,首先要获取到图像文件的存放路径,这里采用标准的文件打开对话框来选取图像文件,文件名存放在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

24位位图在VC++中需要三个结构来存储:

BITMAPFILEHEADER:文件信息头

LPBITMAPINFOHEADER:位图信息头指针

LPBYTE:像素数据指针

假设你的图片名为:“图片.bmp”。

1、你用VC++建立一个MFC(exe)工程,命名为“My”,在d出的“MFC应用程序向导-步骤1”中选“单文档”,然后点“完成”->“确定”。

2、在“MyView.h”文件中找到代码“CMyDoc* GetDocument()”在其下方添加如下代码:

BITMAPINFOHEADER bmih

LPBYTE pBits

BOOL Read(char* s)

3、然后打开“MyView.cpp”文件,在最下面写如下代码:

BOOL CMyView::Read(char* s)

{

CFile file

BITMAPFILEHEADER bmfh

//打开文件

if(!file.Open(s,CFile::modeRead))

{

AfxMessageBox("File cannot open!")

return FALSE

}

//读文件信息头

file.Read( (LPVOID)&bmfh, sizeof(bmfh) )

if(bmfh.bfType != 0x4d42)

{

AfxMessageBox("This is not a bmp file!")

return FALSE

}

//读位图信息头

int infoSize = bmfh.bfOffBits - sizeof(bmfh)

bmih = (LPBITMAPINFOHEADER)new BYTE[infoSize]

file.Read( (LPVOID)bmih, infoSize)

if(bmih->biBitCount!=24)

{

AfxMessageBox("The number of colors is not valid!")

return FALSE

}

//读图像数据

pBits = new BYTE[bmih->biSizeImage]

file.Read( (LPVOID)pBits, bmih->biSizeImage)

return TRUE

}

4、往上找,找到构造函数:CMyView::CMyView()

在其中添加代码:

Read("d:\\picture\\图片.bmp")

5、往下找,找到OnDraw函数,在该函数的第三行添加代码:

if(bmih &&pBits)

{

::StretchDIBits(pDC->GetSafeHdc(),0,0,bmih->biWidth,bmih->biHeight,0,0,bmih->biWidth,bmih->biHeight,pBits,(LPBITMAPINFO)bmih,DIB_RGB_COLORS,SRCCOPY)

}

运行即可。我已试过,可以打开并显示。希望你一步一步照做,代码不要写错!

http://wenku.baidu.com/view/550a8fd049649b6648d7478a.html

http://wenku.baidu.com/view/fe8ff04ffe4733687e21aa95.html

给你个截屏函数:

BOOL GetMyScreen(CDC *pdc // 目标DC)

{

CDC dc

dc.CreateDC("DISPLAY", NULL, NULL, NULL)// 屏幕DC

CRect clientRect

GetClientRect(clientRect) // 对话框矩形区域

pdc->BitBlt(0,0,clientRect.Width(),clientRect.Height(), &dc, 0,0,SRCCOPY )

dc.DeleteDC()

}

看得明白吧,很简单的,往数据库里写,我就不用多说了,呵,有什么不明白的给我发站内。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/sjk/9924721.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-03
下一篇 2023-05-03

发表评论

登录后才能评论

评论列表(0条)

保存