对于数码相机所拍摄出的图片,Exif信息非常重要。Exif是英语Exchangeable Image File(可交换图像文件)的缩写,最初由日本电子工业发展协会(JEIDA --Japan Electronic Industry Development Association) 制订,目前的版本是修改发表于1998年6月的2.1版。国际标准化组织(ISO)正在制订的相机文件设计标准(DCF -- Design role for Camera File system)可能以Exif2.1为基础。当然Exif2.2在2002年也已经出来了,官方网站上也进行了长篇累牍的介绍,但是应用还不够广泛。
Exif 文件实际是JPEG文件的一种,遵从JPEG标准,只是在文件头信息中增加了有关拍摄信息的内容和索引图。所以你可以使用任何支持JPEG格式的图像工具软件观看或修改Exif文件,但,打开时可能看不到Exif信息,一旦修改,Exif信息可能丢失。
很多软件都可以查看文件图片的Exif信息,包括最常用的看图软件ACDSee,但是如何自己获取图片的Exif信息呢?我所要的是把这个功能集成到我的项目中,就必须要自己提取图片的Exif信息。
一些简单的元数据信息,比如创建日期,修改日期,数码相机品牌,型号等等我早已经提取出来,但是对于新的要求,比如光圈值,焦距,ISOSpeed等等,想简单通过文件流读取就没有那么简单了。我参考了一个C#代码,非常冗长,而且不完整,但总算给我提供了一些启示,这是经过略为改造的代码:
using System;
using System.Collections.Generic;
using System.Text;
using System.Drawing;
using System.Drawing.Imaging;
using System.Collections;
using System.ComponentModel;
namespace PictureView
{
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;
}
//曝光时间
case "829a":
{
MyMetadata.ExposureTime.RawValueAsString = BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
//string exposuretime
MyMetadata.ExposureTime.DisplayValue = Value.GetString(MyPropertyItemList[index].Value);
//byte[] a = System.Text.Encoding.ASCII.GetBytes(exposuretime);
//MyMetadata.ExposureTime.DisplayValue = "1/"+a[4].ToString()+"s";
break;
}
case "8827":
{
MyMetadata.ISOSpeed.RawValueAsString = BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
MyMetadata.ISOSpeed.DisplayValue = Value.GetString(MyPropertyItemList[index].Value);
break;
}
case "920a":
{
MyMetadata.FocalLength.RawValueAsString = BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
MyMetadata.FocalLength.DisplayValue = Value.GetString(MyPropertyItemList[index].Value);
break;
}
//光圈值
case "9202":
{
MyMetadata.Aperture.RawValueAsString = BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
MyMetadata.Aperture.DisplayValue = Value.GetString(MyPropertyItemList[index].Value);
break;
}
case "9209":
{
MyMetadata.Flash.RawValueAsString = BitConverter.ToString(MyImage.GetPropertyItem(MyPropertyId).Value);
MyMetadata.Flash.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信息。但是问题又出来了,得到的数据都是ASCII或十六进制值(可以得到一个byte数组),一些数据只要转换为相应的double或int或string类型数据即可得到,但是对我很重要的几个数据,比如焦距(FocalLenth),曝光时间(ExposureTime),光圈值(Aperture)等等,得到的这个ASCII值根本无从转化,经过我查阅了很多资料,才发现:这些数值都是Rational类型,也就是有理数,TIFF是用分数的形式来表达,用了两个LONG类型的数据,前一个LONG为分子,后一个LONG为分母,Size一般也是1。因为一个RATIONAL类型包含两个LONG,无法记录在Value中,所以Value中记录的是这个RATIONAL数所在的位置(从TIFF Header开始的偏移)。我理解为也就是记录的数据的地址。
可是这个偏移量如何使用?TIFF Header的开始值又如何获得?我曾试过多种猜想,事实证明都是不对的。至少不是一个简单的文件中的位置。
网上具体介绍得到这一步的资料非常少,包括Exif官方170几页的技术文档,也没有介绍到这一步应该如何获得,实在是郁闷。不知哪位多媒体高手可以指点迷津。
文章转摘自:http://blog.csdn.net/iorikyo/archive/2006/08/22/1104382.aspx