简介
目前的手机是小米手机,使用两年多里面总共存储了将近4000张照片。拍摄时系统会记录当时拍摄的时间,这个拍摄时间会记录在jpg文件的exif数据结构中,windows系统下右键单击文件可以看到这个时间。但是不知什么原因有一些照片的拍摄时间都变成了2002年12月8日,然而照片文件的命名还是按照当时实际拍摄的时间来命名。而我的手机相册设置又是按照拍摄时间来排序,因此相册的照片排序实在太混乱,后来终于忍受不了决定修改照片文件元数据里的拍摄时间。
照片是jpg格式,内置exif数据结构,Exif按照JPEG的规格在JPEG中插入一些 图像/数字相机 的信息数据以及缩略图像,于是能通过与JPEG兼容的互联网浏览器/图片浏览器/图像处理等一些软件来查看Exif格式的图像文件。就跟浏览通常的JPEG图像文件一样。关于exif的信息和结构可以通过如下两个链接查看:
Exif文件格式描述
EXIF Tags
第一个链接是关于exif的详细介绍,第二个是exif标签的索引,写程序时主要参考的是第二个链接,查找标签比较方便。
OpenExif
在网上找了好久都没找到相关修改照片exif信息的库,很多库都是只能读取exif信息但不能修改。最后只找到C++的OpenExif。本想用python写可能会比较简单,后来只能用麻烦的C++了。
OpenExif是处理exif的一个开源工具,可以从这里下载:OpenExif
OpenExif下载后里面有示例程序,可以方便看到代码是如何运行的。
修改方案
小米手机拍摄的照片都是以IMG开头,命名格式为IMG_YYYYMMDD_HHmmSS。文件命名的时间是照片生成的时间,一般情况下和拍摄时间非常相近。因此只需判断文件名中的时间和元数据中的拍摄时间是否一样,比较时只比较年月日,若不一样就用文件名中的时间替换掉拍摄时间。
因为手机设置了拍照自动记录GPS的功能,因此多做了一步,提取照片时间信息时顺便也提取GPS信息。
代码处理过程
OpenExif有一个操作照片的类ExifImageFile和操作Exif tag的类ExifTagEntry。ExifTagEntry是一个抽象基类。整个过程基本上只用到了这两个类,比较麻烦的是需要知道各类tag的名称。基本的定义和操作如下。
ExifImageFile inImageFile;
ExifTagEntry* genericTagEntry = NULL;
ExifStatus status;
status = inImageFile.open(chImgPath, "r+");
if (status != EXIF_OK)
{
// Error processing...
inImageFile.close();
}
genericTagEntry = inImageFile.getGenericTag(EXIFTAG_DATETIMEORIGINAL, EXIF_APP1_EXIFIFD, status);
if (status != EXIF_OK || NULL == genericTagEntry)
{
//Error processing...
}
else
{
// do something...
}
其中getGenericTag函数的原型是
ExifTagEntry* getGenericTag(exiftag_t tag, ExifTagLocation tagLocation, ExifStatus& errRtn);
拍摄时间的tag名称是EXIFTAG_DATETIMEORIGINAL,而tagLocation可以在ExifTags中查到。
处理的代码如下:
#include "ExifImageDescUtils.h"
#include "ExifImageFile.h"
#include "ExifMisc.h"
#include <io.h>
#include <string>
#include <direct.h>
#include <fstream>
#include <strstream>
#include <sstream>
using namespace std;
// 修改照片的時間
void ModifyMetaTime(vector<string> &PathVector)
{
if(PathVector.size() <= 0)
{
cout<<"No jpg file."<<endl;
return;
}
int nModifyCount = 0; // 记录修改过时间的照片数量
int nTotalCount = 0