检测目标C中PNG文件是否已损坏

时间:2012-05-11 13:18:27

标签: objective-c png nsurlrequest corrupt

我正在使用NSURLRequest下载jpgs和png。这工作正常,但有时文件已损坏。 我见过Catching error: Corrupt JPEG data: premature end of data segment并且这个用于jpgs。 有没有人知道为png做同样的方法?即以编程方式检查png数据是否有效......

3 个答案:

答案 0 :(得分:7)

PNG格式有几个内置检查。每个“块”都有CRC32检查,但要检查您是否需要读取完整文件。

更基本的检查(当然不是万无一失)是读取文件的开头和结尾。

前8个字节应始终为以下(十进制)值{ 137, 80, 78, 71, 13, 10, 26, 10 }ref)。特别是,第二到第四个字节对应于ASCII字符串“PNG”。

十六进制:

89 50 4e 47 0d 0a 1a 0a
.. P  N  G  ...........

您还可以检查文件的最后12个字节(IEND块)。中间4个字节应对应于ASCII字符串“IEND”。更具体地说,最后12个字节应该是(以hexa为单位):

00 00 00 00 49 45 4e 44 ae 42 60 82
........... I  E  N  D  ...........

(严格地说,PNG文件以12个字节结束并不是必须的,IEND块本身就是PNG流的结束信号,因此文件原则上可以有额外的尾随字节,这些字节可以被忽略。 PNG读者。在实践中,这是非常不可能的。)

答案 1 :(得分:2)

就像在Catching error: Corrupt JPEG data: premature end of data segment中一样,这里是PNG的代码段:

- (BOOL)dataIsValidPNG:(NSData *)data
{
    if (!data || data.length < 12)
    {
        return NO;
    }

    NSInteger totalBytes = data.length;
    const char *bytes = (const char *)[data bytes];

    return (bytes[0] == (char)0x89 && // PNG
            bytes[1] == (char)0x50 &&
            bytes[2] == (char)0x4e &&
            bytes[3] == (char)0x47 &&
            bytes[4] == (char)0x0d &&
            bytes[5] == (char)0x0a &&
            bytes[6] == (char)0x1a &&
            bytes[7] == (char)0x0a &&

            bytes[totalBytes - 12] == (char)0x00 && // IEND
            bytes[totalBytes - 11] == (char)0x00 &&
            bytes[totalBytes - 10] == (char)0x00 &&
            bytes[totalBytes - 9] == (char)0x00 &&
            bytes[totalBytes - 8] == (char)0x49 &&
            bytes[totalBytes - 7] == (char)0x45 &&
            bytes[totalBytes - 6] == (char)0x4e &&
            bytes[totalBytes - 5] == (char)0x44 &&
            bytes[totalBytes - 4] == (char)0xae &&
            bytes[totalBytes - 3] == (char)0x42 &&
            bytes[totalBytes - 2] == (char)0x60 &&
            bytes[totalBytes - 1] == (char)0x82);
}

答案 2 :(得分:0)

dataIsValidPNG的更好版本:

BOOL dataIsValidPNG(NSData *data) {

    if (!data) {
        return NO;
    }

    const NSInteger totalBytes = data.length;
    const char *bytes = (const char *)[data bytes];
    const char start[] = { '\x89',  'P',  'N',  'G', '\r', '\n', '\x1a', '\n' };
    const char end[]   = {   '\0', '\0', '\0', '\0',  'I',  'E',    'N',  'D', '\xAE', 'B', '`', '\x82' };

    if (totalBytes < (sizeof(start) + sizeof(end))) {
        return NO;
    }

    return (memcmp(bytes, start, sizeof(start)) == 0) &&
           (memcmp(bytes + (totalBytes - sizeof(end)), end, sizeof(end)) == 0);
}