利用VC# 2005为数码照片添加拍照日期

来源:互联网 发布:知乎 横财命 编辑:程序博客网 时间:2024/04/26 18:41
现在人们具备一台数码相机已经不是什么新鲜事了,更何况500万像素以上的数码相机更是逐渐成了主流。相比较以前以胶卷为感光介质的普通相机,数码相机可 以将所照图像即刻转换成计算机可识别的图像文件格式以便浏览、共享和打印。虽然数码相机在技术和方便性上都远远高于普通相机,但是笔者发现所有已经生成的 图像文件以及打印的数码照片上都没有拍照日期,若想在日后拍照时加上该选项也不是容易事,翻遍了数码相机的说明书竟然没有关于怎样在照片上显示拍照日期的 帮助。而且,随着数码图像文件的不断增加,面对成本成本的相册要想回顾一下其到底是在何时拍照的将会非常困难,尤其像笔者这样不辞辛苦的记录小儿生长历程 的朋友更感觉如此。虽然在桌面电脑上利用我的电脑浏览所拍照的图像文件时,在窗口底部任务栏上或者图像文件属性窗口的摘要页都可以清楚地显示出拍照日期, 但是要想将拍照日期绘制到图像上且能够打印到数码照片上却没有工具可以做到。其实,拍照日期已经保存在了图像文件里,我们需要自己动手编程获取拍照日期并 在图像的右下角将其绘制出来,然后保存新生成的图像文件并拷贝到数码照片打印店进行打印,我们就可以获得具备拍照日期的数码照片了,如下图:


  一、 简介

   目前大部分数码相机都将所拍照的图像保存成JPG格式,而像拍照日期这样的信息统称为EXIF信息。EXIF是英文 ExchangeableImageFile(可交换图像文件)的缩写,最初由日本电子工业发展协会(JEIDA-- JapanElectronicIndustryDevelopmentAssociation)制订,当时JEITA决定为数码相机厂商专门制定一套标 准,随着数码相机的发展,其普及趋势越来越明显,于是JEITA对Exif标准进行了升级,目前最新版本为2.2。其实EXIF也是一种图像文件格式, EXIF信息就是由数码相机在拍摄过程中采集一系列的信息,然后把信息放置在JPG文件或者TIFF文件的头部,也就是说EXIF信息是镶嵌在图像文件格 式内的一组拍摄参数,主要包括摄影时的光圈、快门、ISO、拍照日期时间等各种与当时摄影条件相关的信息、相机品牌型号、色彩编码,甚至还包括拍摄时录制 的声音以及全球定位系统(GPS)等信息。简单的说,它就好像是傻瓜相机的日期打印功能一样,只不过EXIF信息所记录的资讯更为详尽和完备。需要注意的 是,用图像处理软件编辑过的数码图像文件有可能会丢失其EXIF信息。

  所以,要想在图像上绘制拍照日期首先必须读取EXIF信息,然 后将读取出来的拍照日期绘制在图像表面,我们将以500万像素分辨率为2592x1944的JPG图像文件为对象,使用Visual Studio .Net 2005 的C#来编写一工具程序来实现上述功能。

  二、 技术背景

  EXIF信息以键值 对的方式保存在数码JPG图像文件的头部,在.Net平台中所有图像文件头部信息统称为元数据,我们可以使用GDI+读取现有的元数据,也可以将新的元数 据写入图像文件中。GDI+ 将单独的元数据段存储在 PropertyItem 对象中,读取 Image 对象的 PropertyItems 属性以便从某个文件中检索所有的元数据。PropertyItems 属性返回一个 PropertyItem 对象的数组。PropertyItem 对象具有以下四个属性:Id、Value、Len 和 Type。Id用于标识元数据项的标记,下表显示一些Id 的值:

十六进制值 说明 0x0320 图像标题 0x010F 设备制造商 0x0110 设备型号 0x0132 拍照时间 0x829A Exif曝光时间 0x5090 亮度表 .... ....
  Value为数组值,这些值的格式由 Type 属性确定。Len属性指向的值的数组长度(以字节表示)。Type属性指向数组中值的数据类型。下表显示由 Type 属性值指示的格式:

数值 说明 1 一个 Byte 2 ASCII 编码的 Byte 对象的数组 3 16 位整数 4 32 位整数 5 包含两个表示有理数的 Byte 对象的数组 6 未使用 7 未定义 8 未使用 9 SLong 10 SRational
  我们所感兴趣的ID值就是0x0132即图片拍照时间, 对应的标记为PropertyTagDateTime,而在联机的MSDN中我们发现了更详细的关于EXIF属性的GDI+的描述, PropertyTagDateTime值的类型为PropertyTagTypeASCII,它以ASCII编码的形式保存数据,我们在获取数据后就按 照ASCII进行解码,将一些列字节转换为日期/时间的字符串。

  在进行下一步之前,我们可以先用文本编辑软件如UltraEdit打开要操作的图片文件实际看看头文件到底是怎样的,如下图:

点击放大此图片

  我们发现里面的日期格式为:2006:06:07 16:33:41,这个格式既不是标准的日期/时间格式也不是当前系统设置的格式,所以还需要对日期/时间格式进行格式化。

  获得了拍照日期/时间后,从指定的图片文件来创建Graphics对象,在该Graphics对象上绘制先前我们获取的拍照日期/时间。
启动Visual Studio .Net 2005 创建名为PicStamp的Visual C# 项目,选择Windows 应用程序模版。在默认的窗体上放置一个listBox组件用于保存需要绘制拍照日期的图片文件列表,一个textBox组件用于设置绘制后的图片文件所放 置的文件夹,五个Button组件,分别用于向listBox添加图像文件、清空列表框、选择放置绘制后的图片的文件夹、实际绘制操作以及退出示例程序, 一个选择文件对话框用于挑选图片文件,一个选择文件夹对话框用于选择图片文件要放置的文件夹,程序运行界面如下:


  我们自定义一个函数GetExifProperties用于返回图片文件的Exif信息,代码如下:

//获取图像文件的所有元数据属性,以PropertyItem数组的格式保存
public static PropertyItem[] GetExifProperties(string fileName)
{
 FileStream stream = new FileStream(fileName, FileMode.Open, FileAccess.Read);
 //通过指定的数据流来创建Image
 System.Drawing.Image image = System.Drawing.Image.FromStream(stream,true,false);
 return image.PropertyItems;
}
  获得所有元数据后,需要挑选出我们所感兴趣的拍照日期/时间属性所对应的值,代码如下:

//遍历所有元数据,获取拍照日期/时间
private string GetTakePicDateTime(System.Drawing.Imaging.PropertyItem[] parr)
{
 Encoding ascii = Encoding.ASCII ;
 //遍历图像文件元数据,检索所有属性
 foreach (System.Drawing.Imaging.PropertyItem p in parr)
 {
  //如果是PropertyTagDateTime,则返回该属性所对应的值
  if (p.Id==0x0132)
  {
   return ascii.GetString(p.Value);
  }
 }
 //若没有相关的EXIF信息则返回N/A
 return "N/A";
}
  循环处理图片文件列表框中的文件,并重新格式化获取的拍照日期/时间,然后通过Graphics对象将其绘制到数码图像的表面并保存为新文件,代码如下:

……
……
for (int i = 0; i < listBox1.Items.Count; i++)
{
 pi = GetExifProperties(listBox1.Items[i].ToString() );
 //获取元数据中的拍照日期时间,以字符串形式保存
 TakePicDateTime = GetTakePicDateTime(pi);
 //分析字符串分别保存拍照日期和时间的标准格式
 SpaceLocation = TakePicDateTime.IndexOf(" ");
 dt = TakePicDateTime.Substring(0, SpaceLocation);
 dt=dt.Replace(":", "-");
 tm = TakePicDateTime.Substring(SpaceLocation+1, TakePicDateTime.Length - SpaceLocation-2);
 TakePicDateTime = dt + " " + tm;
 //由列表中的文件创建内存位图对象
 Pic = new Bitmap(listBox1.Items[i].ToString());
 //由位图对象创建Graphics对象的实例
 g = Graphics.FromImage(Pic);
 //在Graphics表面绘制数码照片的日期/时间戳
 g.DrawString(TakePicDateTime, normalContentFont, new SolidBrush(normalContentColor),
 Pic.Width - 700, Pic.Height - 200);
 //将添加日期/时间戳后的图像进行保存
 Pic.Save(textBox1.Text + Path.GetFileName(listBox1.Items[i].ToString()));
 //释放内存位图对象
 Pic.Dispose();
}
……
……
  四、 总结

   该程序在Visual Studio .Net 2005 C# + Windows XP SP2下运行成功。通过实际使用该程序可以批量且有效地将数码图片拍照日期/时间绘制到图像表面,我们是以分辨率为2592x1944的JPG图像文件为 绘制对象,读者可以根据实际图片尺寸适当调整源码中拍照日期/时间的字体、大小以及位置。本文仅演示了如何读取EXIF信息,读者可以稍加改动就可以修改 EXIF信息并加以保存。还有需要注意的是,正像本文开头所提到的,任何图像编辑软件对数码照片的编辑都有可能使EXIF信息丢失,本文示例程序也不例 外,经过绘制后的数码图片确实会丢失一些EXIF信息,但是所有关键信息并没有丢失。 
原创粉丝点击