一段时间后 NSData 变为 null

NSData becomes null after a while

目前我正在开发一个 Objective-c / Cocoa 应用程序,可以在其他文档中查找图像。但是我在分析 NSData 时遇到问题,因为一段时间后 NSData 变为空。下面是创建 NSData、获取长度和一些基本调试的代码。

/* Get Data */
NSError *error;
NSData *data = [NSData dataWithContentsOfFile:file options:NSDataReadingMapped error:&error];

/* Split Data into hex chunks */
/* Get length of data */
NSUInteger data_length = [data length];

/* Debug NSLogs */
NSLog(@"[1] Data length: %lu",(unsigned long)data_length);
NSLog(@"[1] Data: %@ with Error: %@",data,error);

当我在此处 NSLog 数据时,传递的文档以其十六进制表示形式写入控制台。

但是,当我尝试处理数据时,最后的调试 NSLog 显示数据为空。 (for 循环检查十六进制 FF 的数据)代码如下:

/* Declare start and end pointer */
NSUInteger start_point = data_length + 2;
NSUInteger end_pointer = data_length + 2;

/* loop through data */
for (NSUInteger i = 0; i < data_length; i++) {
    // Create Range
    NSRange range = NSMakeRange(i, (i+1));
    // Allocate Byte
    Byte byte[1];
    // Get data from data set
    [data getBytes:byte range:range];
    // Convert Byte (char representation) to NSData
    NSData *byte_data = [NSData dataWithBytes:byte length:1];
    // Set string representation of temp_data
    NSString *byte_check = [NSString stringWithFormat:@"%@",byte_data];
    // Check Byte
    if ([byte_check isEqualToString:@"<FF>"]) {
        // First byte found, check if next is correct
        NSRange range2 = NSMakeRange((i+1), (i+2));
        Byte byte2[1];
        [data getBytes:byte2 range:range2];
        NSData *byte_data2 = [NSData dataWithBytes:byte2 length:1];
        NSString *byte_check2 = [NSString stringWithFormat:@"%@",byte_data2];
        if ([byte_check2 isEqualToString:@"<E0>"]) {
            // Success found start of JPEG
            start_point = i;
            // set i to length + 1 to stop loop
            i = data_length + 1;
        }
    }
}

/* Debug point */
NSLog(@"pointer: %lu length: %lu",(unsigned long)end_pointer, (unsigned long)data_length);
NSLog(@"[2] DATA BYTES: %@ with Error: %@",data,error);

在第二部分中,NSData 的长度也变为 0。

下面是 NSLog 输出,为了不让它太长,我缩短了十六进制输出。

2015-09-10 21:11:24.935 PicRecover[51550:13961579] [1] Data length: 66528
2015-09-10 21:11:24.939 PicRecover[51550:13961579] [1] Data: <504b0304 ...
... 01000000> with Error: (null)
2015-09-10 21:11:24.960 PicRecover[51550:13961579] pointer: 0 length: 0
2015-09-10 21:11:24.960 PicRecover[51550:13961579] [2] DATA BYTES: (null) with Error: (null)

希望有人比我更了解NSData,并指出我的错误。如果您需要更多信息,我可以随时添加。

非常感谢您。

NSMakeRange 需要一个偏移量和一个 length,你正在传递两个偏移量......这意味着你的范围比你想象的要大得多,你正在写考虑到您的目标缓冲区只有一个字节长,所有内存都满了。

此外,您对特定字节值的检查过于复杂且不可靠,因为它使用了 description 方法的结果(它在您执行 stringWithFormat: 调用时被调用)。您可以直接比较您的字节值,就像任何其他整数类型一样,例如:

if (byte[0] == 0xFF) ...

最后看看 break 语句的作用。

HTH

要添加到@CRD 的答案中,您应该查看 memmem 和其他现有功能。

此外:您可以使用 CFDataGetBytePtrNSMutableData -mutableBytes.

直接访问内部缓冲区,而不是将 NSData 内容复制到外部缓冲区

对于 NSMakeRange 问题 — C 允许您覆盖任意内存。您只为 byte 变量分配了一个字节,但您正在向该内存位置写入 i + 1 个字节。

这意味着您正在覆盖内存中 byte 旁边的任何信息。

那绝对可以包括您的 data 指针的内容。