1

将 JPEG 图像保存到 iOS 照片库时有一个奇怪的变化。我不知道我是否做错了什么。我正在使用 libjpeg-turbo 访问 JPEG 图像,然后修改图像的 DCT 系数。修改后的图像(仅在 DCT 中,没有别的)保存在照片库中。但是打开保存的图像后,DCT系数与我在上一步中更改的不同。

详细地,让我解释一下我是如何为每个 DCT 添加 +1 的。我正在使用 libjpeg 库的“example.c”中的标准程序:

struct jpeg_decompress_struct cinfo;
struct my_error_mgr jerr;
FILE * infile;

if ((infile = fopen(filename, "rb")) == NULL) {
    fprintf(stderr, "can't open %s\n", filename);
    return 0;
}

cinfo.err = jpeg_std_error(&jerr.pub);
jerr.pub.error_exit = my_error_exit;

if (setjmp(jerr.setjmp_buffer)) {
    jpeg_destroy_decompress(&cinfo);
    fclose(infile);
    return 0;
}

jpeg_create_decompress(&cinfo);

jpeg_stdio_src(&cinfo, infile);

(void) jpeg_read_header(&cinfo, TRUE);

jvirt_barray_ptr* coeffs_array;
coeffs_array = jpeg_read_coefficients(&cinfo);

BOOL done = FALSE;
for (int ci = 0; ci < 3; ci++)
{
    JBLOCKARRAY buffer_one;
    JCOEFPTR blockptr_one;
    jpeg_component_info* compptr_one;
    compptr_one = cinfo.comp_info + ci;

    for (int by = 0; by < compptr_one->height_in_blocks; by++)
    {
        buffer_one = (cinfo.mem->access_virt_barray)((j_common_ptr)&cinfo, coeffs_array[ci], by, (JDIMENSION)1, FALSE);

        for (int bx = 0; bx < compptr_one->width_in_blocks; bx++)
        {
            blockptr_one = buffer_one[0][bx];

            for (int bi = 0; bi < 64; bi++)
            {
                blockptr_one[bi]++;
            }                  
        }   
    } 
}

write_jpeg(output, &cinfo, coeffs_array); // saving modified JPEG to the output file

jpeg_destroy_decompress(&cinfo);
fclose(infile);

在此之后,我在文件中保存了一个新的 JPEG 图像,可以说“new.jpg”。现在我想将这个“new.jpg”保存到照片库中,所以我通过以下方式加载图像:

imageToSave = [UIImage imageWithContentsOfFile:outputFile];

我还检查了 DCT 系数是否保持变化。在我检查了修改后的 DCT 之后,我将保存它:

UIImageWriteToSavedPhotosAlbum(imageToSave, nil, nil, nil);

图像“new.jpg”现在保存在照片库中。

到目前为止,一切都很好,DCT 系数的工作方式与 libjpeg 库一样。当我再次加载保存的图像并查看 DCT 系数时,就会发生变化。我发现 DCT 发生了变化,我不知道为什么。当您要保存 JPEG 图像时,iOS 是否使用了任何优化算法?为什么要更改 DCT。

我正在使用以下过程来读取保存的图像:

NSData *jpegData = UIImageJPEGRepresentation([info objectForKey:UIImagePickerControllerOriginalImage], 1);
[jpegData writeToFile:file atomically:YES]; // saved image is saved in file and I can use the procedure from above to check DCTs

此外,这是原始 DCT 的示例,通过向所有添加 +1 修改的 DCT,以及在将其保存到照片库后为一个 8x8 块加载的 DCT:

Original DCTs:
-759 -24 -8 1 -1 0 0 1 
56 -27 -10 1 0 1 0 0 
8 0 0 0 0 -1 0 0 
0 0 0 -1 0 -1 0 -1 
0 0 0 1 0 0 0 0 
0 0 0 1 0 0 0 0 
0 0 0 0 0 0 0 0 
0 0 0 -1 0 0 0 0 

Modified DCTs by libjpeg:
-758 -23 -7 2 0 1 1 2 
57 -26 -9 2 1 2 1 1 
9 1 1 1 1 0 1 1 
1 1 1 0 1 0 1 0 
1 1 1 2 1 1 1 1 
1 1 1 2 1 1 1 1 
1 1 1 1 1 1 1 1 
1 1 1 0 1 1 1 1 

DCTs after saving JPEG to photo library:
-758 -22 -7 2 0 0 0 0 
58 -26 -8 3 0 0 0 0 
8 2 0 0 -1 1 0 0 
2 3 0 0 0 0 0 0 
2 1 -1 0 0 0 0 0 
0 0 0 0 0 0 0 0 
0 0 0 0 0 0 0 0 
0 0 0 0 0 0 0 -1 

任何帮助,将不胜感激。我真的不知道为什么在将图像保存到照片库后会更改 DCT。

4

2 回答 2

2

当您使用 UIImage 及其方法时,iOS 将始终重新压缩您的 JPEG,从而改变 DCT。相反,使用 AssetsLibrary 将用户的图片存储和检索为 NSData。此行为未记录在案,但它有效。

我还建议将 .jpeg 存储在临时文件夹中,然后将其提供给 libjpeg-turbo。

于 2012-12-26T23:41:42.140 回答
-1

您应该明确地简化您的代码,以查看不需要的行为发生在哪里。您正在执行至少五个可能出错的不同步骤。

现在,也就是说,不在 iOS 上工作但知道 JPEG 编解码器,请注意用于保存file.jpg的质量。我不确定您的特定库的默认值是什么,但如果不是 100%,它肯定会影响您的 DCT 系数。

尝试使用无损编解码器(例如 PNG 或 BMP)保存修改后的图像。

于 2019-02-04T15:30:31.613 回答