
本文旨在介绍如何在 PHP 中读取和写入 WebP 图像的元数据,包括 EXIF 和 XMP 数据。WebP 格式本身支持这些元数据,但 PHP 的 `exif_read_data` 函数可能无法直接读取。本文将提供一种通过直接操作 WebP 文件结构的方式来添加元数据的方法,并提供示例代码。
WebP 格式与元数据支持
WebP 图像格式基于 RIFF (Resource Interchange File Format) 容器格式,允许嵌入多种类型的元数据,包括 EXIF、XMP 和 ICCP。这意味着 WebP 本身是支持元数据的存储的。
PHP exif_read_data 函数的局限性
PHP 的 exif_read_data 函数在处理 WebP 图像时,可能会因为底层库的支持问题而无法正确读取元数据。这并不意味着 WebP 不支持元数据,而是 PHP 的函数可能存在兼容性问题。
手动添加元数据到 WebP 文件
由于 exif_read_data 函数的局限性,我们可以通过直接操作 WebP 文件的结构来添加元数据。WebP 文件由多个 Chunk 组成,每个 Chunk 包含一个 4 字节的标识符(FourCC),一个 4 字节的 Chunk 大小,以及实际的 Chunk 数据。
立即学习“PHP免费学习笔记(深入)”;
WebP 文件的基本结构如下:
-
RIFF 头部:
- RIFF (4 字节): 文件标识符
- 文件大小 (4 字节): 整个文件的大小减去 8 字节
- WEBP (4 字节): WebP 格式标识符
-
Chunk 数据:
- Chunk ID (4 字节): 例如 EXIF、XMP 或 ICCP
- Chunk 大小 (4 字节): Chunk 数据的长度
- Chunk 数据 (可变长度): 实际的元数据内容
要添加元数据,我们需要创建一个新的 Chunk,将其附加到 WebP 文件的末尾,并更新文件大小。
PHP 示例代码:添加 EXIF 数据
以下是一个 PHP 示例代码,演示如何将 EXIF 数据添加到 WebP 文件中:
<?php
/**
* Adds EXIF metadata to a WebP image file.
*
* @param string $targetFile The path to the WebP image file.
* @param string $exifData The binary EXIF data to add.
* @return bool True on success, false on failure.
*/
function addExifToWebP(string $targetFile, string $exifData): bool
{
$exifLength = strlen($exifData);
// RIFF requires 16-bit alignment
if ($exifLength % 2 == 1) {
$exifData .= "\0";
$exifLength++; // Update length after padding
}
$fileHandle = fopen($targetFile, 'r+');
if (!$fileHandle) {
return false; // Failed to open file for writing
}
fseek($fileHandle, 0, SEEK_END); // Go to end of file
// Write EXIF chunk
fwrite($fileHandle, 'EXIF'); // 4 bytes chunk ID
fwrite($fileHandle, pack('V', $exifLength)); // 4 bytes of payload length
fwrite($fileHandle, $exifData); // Actual data
$fileSize = ftell($fileHandle); // Get new file size
fseek($fileHandle, 4, SEEK_SET); // Go to 5th byte of file
fwrite($fileHandle, pack('V', $fileSize - 8)); // Write 4 bytes, patching old filesize
fclose($fileHandle); // Store everything
return true;
}
// Example usage:
$targetWebP = 'target.webp';
$exifData = file_get_contents('source.jpg'); // Get EXIF data from a JPEG file
// Extract EXIF data from JPG using exif_read_data
$exif = exif_read_data('source.jpg');
if ($exif === false) {
echo "No EXIF data found in source.jpg.\n";
} else {
// Convert the EXIF data to a binary string (this is a simplified example)
$exifData = serialize($exif); // Consider using a proper EXIF serialization library
if (addExifToWebP($targetWebP, $exifData)) {
echo "EXIF data added to $targetWebP successfully.\n";
} else {
echo "Failed to add EXIF data to $targetWebP.\n";
}
}
?>代码解释:
-
addExifToWebP 函数:
- 接受 WebP 文件路径和 EXIF 数据作为参数。
- 打开 WebP 文件,并将文件指针移动到文件末尾。
- 写入 EXIF Chunk 标识符、EXIF 数据长度和实际的 EXIF 数据。
- 获取新的文件大小,并更新文件头部的文件大小信息。
- 关闭文件。
-
示例用法:
- 从 JPEG 文件读取 EXIF 数据(需要确保 source.jpg 存在)。
- 调用 addExifToWebP 函数将 EXIF 数据添加到 WebP 文件。
注意事项:
- 该示例使用了 serialize 函数来将 EXIF 数组转换为字符串。在实际应用中,建议使用专门的 EXIF 序列化库,以确保数据的正确性和兼容性。
- RIFF 格式要求 Chunk 数据进行 16 位对齐。如果 Chunk 数据长度为奇数,需要在末尾添加一个空字节 \0。
- 错误处理:代码包含基本的错误处理,例如检查文件是否成功打开。在生产环境中,应该添加更完善的错误处理机制。
总结
虽然 PHP 的 exif_read_data 函数可能无法直接读取 WebP 图像的元数据,但 WebP 格式本身支持元数据存储。通过手动操作 WebP 文件结构,我们可以将 EXIF 或 XMP 数据添加到 WebP 文件中。请务必注意文件格式的细节,并进行适当的错误处理。











