有些特殊的需求需要批量修改图片的md5哈希值,md5是文件的唯一标示,它是根据文件的所有字节运算得来的。只要文件数据有变化,md5就会改变。
1 方案
图片的数据分为两部分,图片内容和描述信息,内容是二进制的,描述信息是文本类型,可以读取和修改。用16进制文本编辑器查看图片:
修改图片md5也就有两种方式:
如果直接修改图片内容的二进制,可能会损坏图片。如果用图片工具修改图片的像素,不会损坏图片,但是效率比较低,也不能保证修改后没有视觉差异。这种方式不可行。
如果修改图片的描述信息,不会对内容有任何影响。也能保证md5值改变。
2 工具
2.1 编辑jpg和png图片
exiftool https://exiftool.org/
exiftool是一个命令行工具,可以读取和修改几乎所有图片格式的描述信息,说“几乎”是因为它不能修改webp图片(webp是一种体积很小,使用较多的图片格式)。
按照官方说明安装好之后,使用非常简单,命令行输入
exiftool a.png
就可以获取图片的exif信息了:
写入信息类似,比如要写入一个comment
exiftool -comment="1234567890" a.png
查看可以看到comment已经写进去了。比较写入前后图片md5,发现已经改变了。
2.2 编辑webp图片
https://developers.google.com/speed/webp
编辑webp的exif信息需要使用Google官方的工具,安装好命令行工具后,使用webpmux命令
Add EXIF metadata: 添加exif信息
webpmux -set exif image_metadata.exif in.webp -o exif_container.webp
Extract EXIF metadata: 读取exif信息
webpmux -get exif exif_container.webp -o image_metadata.exif
添加exif信息需要提供image_metadata.exif文件,(这个格式的文件没有见过,找了很多webp文件读取也没有读出来)。
新建一个文本文件a.exif,在里面输入"1234567890"(任意字符串),用如下命令都可以写进图片。
webpmux -set exif a.exif a.webp -o a.webp
webpmux -get exif a.webp -o b.exif
将exif信息读取到b.exif文件中,内容跟写进a.exif的一致。
3 批处理
使用Java调用命令行就可以进行批量处理了。
Java执行命令行:
Runtime.getRuntime().exec(exec)
递归处理文件夹
/**
* 递归修改图片exif信息
* @param path 文件或文件夹路径
* @param comment 写入的数据
* @param commentFile 包含写入数据的exif文件
*/
public static void editExifRecurve(String path, String comment, String commentFile) {
File file = new File(path);
if (file.isDirectory()) {
File[] files = file.listFiles();
if (files != null) {
log("dir " + file.getAbsolutePath());
for (File f : files) {
editExifRecurve(f.getAbsolutePath(), comment, commentFile);
}
}
} else {
editExif(path, comment, commentFile);
}
}
public static void editExif(String path, String comment, String commentFile) {
String low = path.toLowerCase();
String exec = "";
if (low.endsWith(".jpg") || low.endsWith(".jpeg") || low.endsWith(".png")) {
exec = "exiftool -comment=" + comment + " " + path;
} else if(low.endsWith(".webp")) {
exec = "webpmux -set exif " + commentFile + " " + path + " -o " + path;
} else {
return;
}
// File file = new File(path);
// String md5Old = MessageDigestUtil.getFileMD5String(file);
try {
Runtime.getRuntime().exec(exec);
// Runtime.getRuntime().exec(exec).waitFor();
} catch (Exception e) {
e.printStackTrace();
}
// String md5New = MessageDigestUtil.getFileMD5String(file);
// String res = md5New.equals(md5Old) ? "x" : "√";
// log(res + " md5 " + md5Old + " -> " + md5New + " " + file.getName());
}