一段时间后 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
和其他现有功能。
此外:您可以使用 CFDataGetBytePtr
或 NSMutableData -mutableBytes
.
直接访问内部缓冲区,而不是将 NSData
内容复制到外部缓冲区
对于 NSMakeRange
问题 — C 允许您覆盖任意内存。您只为 byte
变量分配了一个字节,但您正在向该内存位置写入 i + 1
个字节。
这意味着您正在覆盖内存中 byte
旁边的任何信息。
那绝对可以包括您的 data
指针的内容。
目前我正在开发一个 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
和其他现有功能。
此外:您可以使用 CFDataGetBytePtr
或 NSMutableData -mutableBytes
.
NSData
内容复制到外部缓冲区
对于 NSMakeRange
问题 — C 允许您覆盖任意内存。您只为 byte
变量分配了一个字节,但您正在向该内存位置写入 i + 1
个字节。
这意味着您正在覆盖内存中 byte
旁边的任何信息。
那绝对可以包括您的 data
指针的内容。