简介

目前的手机是小米手机,使用两年多里面总共存储了将近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; // 小米手机拍的照片数量,IMG开头ofstream                    file;ExifImageFile               inImageFile;ExifTagEntry*               genericTagEntry = NULL;ExifStatus                  status;vector<string>::iterator    iter = PathVector.begin();file.open("K:\\00-Mobile Phone Pics Test\\modifylog.txt", ios::out | ios::_Noreplace | ios::trunc);cout << "OriginalTime" << "         " << "ModifiedTime" << endl;file << "FileName,OriginalTime,ModifyedTime,status" << endl;while (iter != PathVector.end()){       string str = *iter;const char* chPath = str.c_str();// 目录格式固定:K:\00-Mobile Phone Pics Testchar chTag[4] = "\0";strncpy_s(chTag, chPath+29, 3);// 不是本手机的照片if (strcmp(chTag, "IMG") != 0){iter++;continue;}nTotalCount++;char chYear[6] = "\0";char chMonth[4] = "\0";char chDay[4] = "\0";char chHour[4] = "\0";char chMinute[4] = "\0";char chSecond[4] = "\0";char chTime[20] = "\0";strncpy_s(chYear, chPath+33, 4);strncpy_s(chMonth, chPath+37, 2);strncpy_s(chDay, chPath+39, 2);strncpy_s(chHour, chPath+42, 2);strncpy_s(chMinute, chPath+44, 2);strncpy_s(chSecond, chPath+46, 2);strncat_s(chTime, chYear, 4);strncat_s(chTime, ":", 1);strncat_s(chTime, chMonth, 2);strncat_s(chTime, ":", 1);strncat_s(chTime, chDay, 2);strncat_s(chTime, " ", 1);strncat_s(chTime, chHour, 2);strncat_s(chTime, ":", 1);strncat_s(chTime, chMinute, 2);strncat_s(chTime, ":", 1);strncat_s(chTime, chSecond, 2);ExifStatus openStatus = inImageFile.open(chPath, "r+");if (openStatus != EXIF_OK) // 不能用w打开照片,打开后会重新创建一张照片并将原来的照片覆盖掉{cout << "Error: could not open " << chPath << endl;file << *(iter) << " , ," << "1" << endl;iter++;inImageFile.close();continue;}// 用getGenericTag函数获取指定的时间信息// 然后实例化一个ExifTagEntryT,用setGenericTag设置时间// 定义一个基类指针指向ExifTagEntryT的对象,然后该指针就可以调用setValue// 之后用setGenericTag设置时间genericTagEntry = inImageFile.getGenericTag(EXIFTAG_DATETIMEORIGINAL, EXIF_APP1_EXIFIFD, status);if (status != EXIF_OK || NULL == genericTagEntry) // exif信息有误,重新赋值{genericTagEntry = new ExifTagEntryT<std::string>(EXIFTAG_DATETIMEORIGINAL, EXIF_ASCII, 20, chTime);if (inImageFile.setGenericTag(*genericTagEntry, EXIF_APP1_EXIFIFD) != EXIF_OK){cout << *(iter);}else{cout << chTime << endl;file << *(iter) << ", ," << chTime << ",0" << endl;}inImageFile.close();nModifyCount++;     iter++;continue;}std::string OriginalTime = ((ExifTagEntryT<std::string>*)genericTagEntry)->getValue();// 文件名的时间和拍摄时间可能不一样,文件名的时间可能比拍摄时间短,因此只比较年月日// 如果时间不一样就用文件名的时间代替拍摄时间char metaTimeYMD[16] = "\0";char metaTime[20] = "\0";if (OriginalTime.length() > 0){strcpy(metaTime, OriginalTime.c_str());     strncpy_s(metaTimeYMD, metaTime, 10);}char fileTime[16] = "\0";strncpy_s(fileTime, chTime, 10);if (strcmp(metaTimeYMD, fileTime) != 0){genericTagEntry = new ExifTagEntryT<std::string>(EXIFTAG_DATETIMEORIGINAL, EXIF_ASCII, 20, chTime);inImageFile.setGenericTag(*genericTagEntry, EXIF_APP1_EXIFIFD);inImageFile.close();nModifyCount++;cout << metaTime << "  " << chTime << endl;file << *(iter) << "," << metaTime << "," << chTime << "," << endl;}inImageFile.close();iter++;}cout << nTotalCount << " out of " << PathVector.size() << " pictrues have been chosen." << endl;cout << nModifyCount << " out of " << nTotalCount << " chosen pictures have been modified." << endl;file.close();}// 获取需要修改的照片路径列表
vector<string> ModifyPicsTime(void)
{char PicDir[] = "K:\\00-Mobile Phone Pics Test\\";vector<string> PathVector;PathVector.clear();if (_access(PicDir, 06) == -1){cout << "Directory does not exit" << endl;return PathVector;}if (_chdir(PicDir) != 0){cout << "Can't change the work dirctroy." << endl;return PathVector;}_finddata_t fileinfo;memset(&fileinfo, 0x0, sizeof(fileinfo));intptr_t iFind = _findfirst("*.jpg", &fileinfo);if (iFind == -1){cout << "Can't find any jpg file." << endl;return PathVector;}char jpgFile[512] = "\0";strcat_s(jpgFile, PicDir);strcat_s(jpgFile, fileinfo.name);string str;str = jpgFile;PathVector.push_back(str);while (_findnext(iFind, &fileinfo) == 0){memset(jpgFile, 0, 512);strcat_s(jpgFile, PicDir);strcat_s(jpgFile, fileinfo.name);str = jpgFile;PathVector.push_back(str);}return PathVector;
}// 获取GPS信息并将其写入文本文件中
void GetGPSInfo(void)
{vector<string> PathVector = ModifyPicsTime();if (PathVector.size() <= 0){cout << "No jpg file." << endl;return;}int                         nCount = 0;ofstream                    file;ExifImageFile               inImageFile;ExifTagEntry*               genericTagEntry = NULL;ExifStatus                  status;vector<string>::iterator    iter = PathVector.begin();file.open("K:\\00-Mobile Phone Pics Test\\GPSInfo.txt", ios::out | ios::_Noreplace | ios::trunc);while (iter != PathVector.end()){string str = *iter;const char* chPath = str.c_str();ExifStatus openStatus = inImageFile.open(chPath, "r+");if (openStatus != EXIF_OK) // 不能用w打开照片,打开后会重新创建一张照片并将原来的照片覆盖掉{iter++;inImageFile.close();continue;}genericTagEntry = inImageFile.getGenericTag(EXIFTAG_GPSLATITUDE, EXIF_APP1_IFD0_GPSINFO_IFD, status);if (status != EXIF_OK || NULL == genericTagEntry) {iter++;inImageFile.close();continue;}std::vector<float> fLatitudeArray = ((ExifTagEntryT<std::vector<float>>*)genericTagEntry)->getValue();float fLatitude = fLatitudeArray[0] + fLatitudeArray[1] / 60.0 + fLatitudeArray[2] / 3600.0;genericTagEntry = inImageFile.getGenericTag(EXIFTAG_GPSLONGITUDE, EXIF_APP1_IFD0_GPSINFO_IFD, status);if (status != EXIF_OK || NULL == genericTagEntry) {iter++;inImageFile.close();continue;}std::vector<float> fLongitudeArray = ((ExifTagEntryT<std::vector<float>>*)genericTagEntry)->getValue();float fLongitude = fLongitudeArray[0] + fLongitudeArray[1] / 60.0 + fLongitudeArray[2] / 3600.0;// 取時間std::string OriginalTime = " ";genericTagEntry = inImageFile.getGenericTag(EXIFTAG_DATETIMEORIGINAL, EXIF_APP1_EXIFIFD, status);if (status == EXIF_OK && NULL != genericTagEntry){OriginalTime = ((ExifTagEntryT<std::string>*)genericTagEntry)->getValue();}nCount++;cout << fLatitude << "," << fLongitude << "," << OriginalTime << "," << nCount << endl;file << fLatitude << "," << fLongitude << "," << OriginalTime << endl;iter++;inImageFile.close();}file.close();cout << "total pics: " << nCount << endl;
}// Main
int main(int argc, char* argv[])
{// 获取GPS信息GetGPSInfo();// 取得要修改的照片路径列表vector<string> PathVector = ModifyPicsTime();// 修改拍摄时间ModifyMetaTime(PathVector);return 0;
}

最后终于将绝大部分照片的时间都改了回来,只有几张因为当时拍照时刚好将SIM卡拔出,导致系统时间为1970年,使得拍摄的照片文件名和元数据显示的都是1970年,未能更改。最后的输出结果如下:

使用OpenExif修改jpeg图片信息相关推荐

  1. 使用libexif开源库修改jpeg相片exif信息

    使用libexif开源库修改jpeg相片exif信息 libexif简介 一.读exif信息 二.写exif信息 说明 例:修改exif中GPS海拔高度 libexif简介 libexif是一个开源的 ...

  2. 微信小程序轮播图单独添加图片、修改轮播图图片、单独修改某张图片

    小老弟上课的基本见解,有错误欢迎大牛们指正 <!--pages/swiper/swiper.wxml--> <text>pages/swiper/swiper.wxml< ...

  3. Efficient reversible data hiding for JPEG images with multiple histograms modification(论文阅读)

    目录 论文背景 论文贡献 基于未压缩图片的可逆数据隐藏 现有的基于JEPG图像的可逆数据隐藏方法 论文提出的方法 一般论文都是先提出问题后解决问题,所以在阅读一篇论文的时候要清楚论文要解决的问题是什么 ...

  4. 【STM32F429开发板用户手册】第46章 STM32F429的DMA2D应用之刷色块,位图和Alpha混合

    最新教程下载:http://www.armbbs.cn/forum.php?mod=viewthread&tid=93255 第46章       STM32F429的DMA2D应用之刷色块, ...

  5. 会议OA项目(我的会议中的会议排座送审功能)

    文章目录 一.会议排座插件介绍 1)会议项目为什么要有会议排座的功能 2)完成在页面上元素的拖动功能 2.1分析现有素材的不足 2.2修改现有素材的不足⬇⬇⬇ 2.3 content需要传递到后台,并 ...

  6. 会议OA(会议排座送审)

    目录 一,会议排座 二,会议送审 一,会议排座 分析: 1. 查找资料     做选择,哪一个素材更适合完成需求 2. 素材改造     素材的缺陷:         ①:样式:座位小方块重叠/太小 ...

  7. Bugku-杂项部分题目WP

    前言 一个新手学习CTFer的成长之路必定要通过大量的刷题,下面是bugku部分题目wp Bugku-MISC-论剑 把图片放进010发现有一串二进制先转ASCII码获得一个关键词,但不知道是什么东西 ...

  8. Hybrid LSTM and Encoder–Decoder Architecture for Detection of Image Forgeries论文阅读

    Hybrid LSTM and Encoder–Decoder Architecture for Detection of Image Forgeries Code link:https://gith ...

  9. python 图标复制(Turtle)

    python-复制图标 课题要求 1.给定一个图标,如校徽,识别校徽的轮廓.颜色.内容等特征: 2.利用turtle工具包演示该校徽绘制的全过程,并将新生成图标保存为png图片. 需求分析 1. 本课 ...

最新文章

  1. 使用tomcat自带的连接池,报错
  2. MySQL注入中报错的利用
  3. 硅谷与人工智能的一段风流暧昧史
  4. java判断五张牌中有一对,同花大顺-扑克牌问题一副扑克牌中任意取出五张牌,那五 – 手机爱问...
  5. VISIO2010界面介绍
  6. 将数据导入excel表格
  7. 深入理解mysql百度网_深入理解mysql
  8. 爬了知乎 200 万数据,图说程序员都喜欢去哪儿工作
  9. VC++6遇到的问题(持续更新)
  10. ZOJ4037 Peer Review
  11. 前端知识点查文档网站
  12. html右键菜单背景图片,win10系统设置鼠标右键菜单背景图片的步骤
  13. 数值分析与算法——读书笔记(一)
  14. c语言中英文字幕怎么相加,excel表格怎么快速分离中英文双语字幕? excel分离中英文字符串的技巧...
  15. 股票量化交易进阶001_回测框架backtrader(一)
  16. 【Ubuntu】Ubuntu20.04安装GPU显卡驱动
  17. 记一道80%的人会答错的牛X面试题!
  18. 2023最新SSM计算机毕业设计选题大全(附源码+LW)之java自助旅游平台v294n
  19. LBS-手机定位应用
  20. PKCS1_SignatureScheme_PSS

热门文章

  1. postgres主从配置
  2. (原創) 如何将字符串前后的空白去除? (使用string.find_first_not_of, string.find_last_not_of) (C/C++)...
  3. Element UI自定义表单验证 公共提取
  4. Linux运维常见问题解决集锦【转】
  5. playframework文档未提及,但你能做的事
  6. mongodb 总结
  7. 面试:高并发系统设计
  8. 在 npm script 中使用环境变量
  9. Oracle RMAN备份与还原 - 联机备份讲解
  10. Spring Cloud CLI简介