如何在 C 中编辑 EXIF 标签?

Posted

技术标签:

【中文标题】如何在 C 中编辑 EXIF 标签?【英文标题】:How to edit EXIF tag in C? 【发布时间】:2013-01-15 04:49:47 【问题描述】:

我有一个使用 BlackBerry 10 Dev Alpha 拍摄的 JPG 文件。此代码(this example 的略微修改版本)正确打印结果

static char* read_tag(ExifData *ed, ExifIfd eid, ExifTag tag)
static char result[1024];
ExifEntry *entry = exif_content_get_entry(ed->ifd[eid], tag);

if (entry)
    char buf[1024];

    exif_entry_get_value(entry, buf, sizeof(buf));
    trim_spaces(buf);

    if (*buf) strcpy(result, buf);
    else strcpy(result, "NULL");

else strcpy(result, "NULL");

return result;

这意味着输出:

printf("Model : %s\n", read_tag(ed, EXIF_IFD_0, EXIF_TAG_MODEL));

是:

型号:BlackBerry 10 Dev Alpha

现在我想知道如何将“BlackBerry 10 Dev Alpha”(EXIF_TAG_MODEL)替换为另一个值,例如“Nokia 3330”。我已经看过another example 。不幸的是,我发现它很难阅读。也许有人有更短/简单的代码?

【问题讨论】:

链接示例的哪个部分您觉得难以阅读?如果您提供一些行号,我会看看我能做什么。 我猜是从第 263 行开始。但这当然是关于编写一个新标签。我想要的是编辑现有标签。 【参考方案1】:

libexif 不支持直接加载 JPG。您需要另一个包来读取 JPG 并提取 EXIF 标头(或者您可以自己编写一些东西)。

请注意,在示例中,它只是创建了一个新的 exif 标头,然后使用 fwrite 将其保存到文件中,然后在此处的这部分代码的末尾附加了没有 exif 信息的原始 JPG 数据:

 /* Write JPEG image data, skipping the non-EXIF header */
            if (fwrite(image_jpg+image_data_offset, image_data_len, 1, f) != 1) 
                    fprintf(stderr, "Error writing to file %s\n", FILE_NAME);
                    goto errout;
            

有一个名为 exifyay 的优秀 Github 项目,它使用 libexif 并有两个额外的库来处理 JPGS 中的读取。这是一个 python 项目,但库的源代码是 C。你可以找到 exifyay here(注意我与 exifyay 或 libexif 没有任何关系)

我最近刚刚编译了 libexif 并将 exifyay 的源代码合并到一个 VS2010 项目 here 中。文件夹“contrib\examples\LibexifExample”中有一个示例。如果你不喜欢下载随机链接,这里是我工作的代码示例:

/*
 * write-exif.c
 *
 * Placed into the public domain by Daniel Fandrich
 *
 * Create a new EXIF data block and write it into a JPEG image file.
 *
 * The JPEG image data used in this example is fixed and is guaranteed not
 * to contain an EXIF tag block already, so it is easy to precompute where
 * in the file the EXIF data should be. In real life, a library like
 * libjpeg (included with the exif command-line tool source code) would
 * be used to write to an existing JPEG file.
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <libexif/exif-data.h>
#include <libjpeg/jpeg-data.h>
#include <JpegEncoderEXIF/JpegEncoderEXIF.h>


/* byte order to use in the EXIF block */
#define FILE_BYTE_ORDER EXIF_BYTE_ORDER_INTEL

/* comment to write into the EXIF block */
#define FILE_COMMENT "libexif demonstration image"

/* special header required for EXIF_TAG_USER_COMMENT */
#define ASCII_COMMENT "ASCII\0\0\0"

static ExifEntry *create_tag(ExifData *exif, ExifIfd ifd, ExifTag tag, size_t len)

    void *buf;
    ExifEntry *entry;

    /* Create a memory allocator to manage this ExifEntry */
    ExifMem *mem = exif_mem_new_default();
    assert(mem != NULL); /* catch an out of memory condition */

    /* Create a new ExifEntry using our allocator */
    entry = exif_entry_new_mem (mem);
    assert(entry != NULL);

    /* Allocate memory to use for holding the tag data */
    buf = exif_mem_alloc(mem, len);
    assert(buf != NULL);

    /* Fill in the entry */
    entry->data = (unsigned char*)buf;
    entry->size = len;
    entry->tag = tag;
    entry->components = len;
    entry->format = EXIF_FORMAT_UNDEFINED;

    /* Attach the ExifEntry to an IFD */
    exif_content_add_entry (exif->ifd[ifd], entry);

    /* The ExifMem and ExifEntry are now owned elsewhere */
    exif_mem_unref(mem);
    exif_entry_unref(entry);

    return entry;


int main(int argc, char **argv)


    ExifEntry *entry;

    //Input JPG
    char mInputFilename[]="example.jpg";

    //Load JPG
    JPEGData * mJpegData = jpeg_data_new_from_file(mInputFilename);

    //Load Exif data from JPG
    ExifData * mExifData = jpeg_data_get_exif_data(mJpegData);

    //Set some Exif options
    exif_data_set_option(mExifData, EXIF_DATA_OPTION_FOLLOW_SPECIFICATION);
    exif_data_set_data_type(mExifData, EXIF_DATA_TYPE_COMPRESSED);
    exif_data_set_byte_order(mExifData, FILE_BYTE_ORDER);

    entry = create_tag(mExifData, EXIF_IFD_EXIF, EXIF_TAG_USER_COMMENT, 
            sizeof(ASCII_COMMENT) + sizeof(FILE_COMMENT) - 2);
    /* Write the special header needed for a comment tag */
    memcpy(entry->data, ASCII_COMMENT, sizeof(ASCII_COMMENT)-1);
    /* Write the actual comment text, without the trailing NUL character */
    memcpy(entry->data+8, FILE_COMMENT, sizeof(FILE_COMMENT)-1);
    /* create_tag() happens to set the format and components correctly for
     * EXIF_TAG_USER_COMMENT, so there is nothing more to do. */

    /* Create a EXIF_TAG_SUBJECT_AREA tag */
    entry = create_tag(mExifData, EXIF_IFD_EXIF, EXIF_TAG_SUBJECT_AREA,
               4 * exif_format_get_size(EXIF_FORMAT_SHORT));
    entry->format = EXIF_FORMAT_SHORT;
    entry->components = 4;

    //Write back exif data
    jpeg_data_set_exif_data(mJpegData,mExifData);

    //Save to JPG
    jpeg_data_save_file(mJpegData,"test.jpg");

    return 0;

【讨论】:

"libexif 不支持直接加载 JPG" - exif_data_new_from_file() 函数正是这样做的。

以上是关于如何在 C 中编辑 EXIF 标签?的主要内容,如果未能解决你的问题,请参考以下文章

如何在Android中将exif数据写入图像?

是啥让一些 EXIF 标签不可写?

如何通过 graphicsmagick 命令编辑 exif 元数据?

Libexif ,附加新的 exif 数据

使用 libexif 和 libjpeg 在现有 JPEG 上设置 exif 标签

如何使用 IPTC/EXIF 元数据对照片进行分类?