有些特殊的需求需要批量修改圖片的md5哈希值,md5是文件的唯一標示,它是根據(jù)文件的所有字節(jié)運算得來的。只要文件數(shù)據(jù)有變化,md5就會改變。
1 方案
圖片的數(shù)據(jù)分為兩部分,圖片內(nèi)容和描述信息,內(nèi)容是二進制的,描述信息是文本類型,可以讀取和修改。用16進制文本編輯器查看圖片:

修改圖片md5也就有兩種方式:
如果直接修改圖片內(nèi)容的二進制,可能會損壞圖片。如果用圖片工具修改圖片的像素,不會損壞圖片,但是效率比較低,也不能保證修改后沒有視覺差異。這種方式不可行。
如果修改圖片的描述信息,不會對內(nèi)容有任何影響。也能保證md5值改變。
2 工具
2.1 編輯jpg和png圖片
exiftool https://exiftool.org/
exiftool是一個命令行工具,可以讀取和修改幾乎所有圖片格式的描述信息,說“幾乎”是因為它不能修改webp圖片(webp是一種體積很小,使用較多的圖片格式)。
按照官方說明安裝好之后,使用非常簡單,命令行輸入
exiftool a.png
就可以獲取圖片的exif信息了:

寫入信息類似,比如要寫入一個comment
exiftool -comment="1234567890" a.png

查看可以看到comment已經(jīng)寫進去了。比較寫入前后圖片md5,發(fā)現(xiàn)已經(jīng)改變了。
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文件中,內(nèi)容跟寫進a.exif的一致。
3 批處理
使用Java調(diào)用命令行就可以進行批量處理了。
Java執(zhí)行命令行:
Runtime.getRuntime().exec(exec)
遞歸處理文件夾
/**
* 遞歸修改圖片exif信息
* @param path 文件或文件夾路徑
* @param comment 寫入的數(shù)據(jù)
* @param commentFile 包含寫入數(shù)據(jù)的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());
}