5

参考以下问题:Convert NSData into HEX NSString

我已经使用 Erik Aigner 提供的解决方案解决了这个问题,即:

NSData *data = ...;
NSUInteger capacity = [data length] * 2;
NSMutableString *stringBuffer = [NSMutableString stringWithCapacity:capacity];
const unsigned char *dataBuffer = [data bytes];
NSInteger i;
for (i=0; i<[data length]; ++i) {
  [stringBuffer appendFormat:@"%02X", (NSUInteger)dataBuffer[i]];
}

但是,有一个小问题,如果后面有多余的零,则字符串值会有所不同。例如。如果六进制数据是字符串@“3700000000000000”,则在使用扫描仪转换为整数时:

unsigned result = 0;
NSScanner *scanner = [NSScanner scannerWithString:stringBuffer];
[scanner scanHexInt:&result];
NSLog(@"INTEGER: %u",result);

结果将是 4294967295,这是不正确的。不应该是 55,因为只取六边形 37 吗?

那么如何摆脱零点呢?

编辑:(响应 CRD)

嗨,感谢您澄清我的疑问。所以你正在做的是直接从字节指针中读取 64 位整数,对吗?不过我还有一个问题。您实际上如何将 NSData 转换为字节指针?

为了让您更容易理解,我将解释我最初所做的事情。

首先,我显示了我拥有的文件的数据(数据是十六进制的)

NSData *file = [NSData dataWithContentsOfFile:@"file path here"];
NSLog(@"Patch File: %@",file);

输出:

在此处输入图像描述

接下来,我读取并偏移文件的前 8 个字节,并将它们转换为字符串。

// 0-8 bytes
[file seekToFileOffset:0];
NSData *b = [file readDataOfLength:8];
NSUInteger capacity = [b length] * 2;
NSMutableString *stringBuffer = [NSMutableString stringWithCapacity:capacity];
const unsigned char *dataBuffer = [b bytes];
NSInteger i;
for (i=0; i<[b length]; ++i) {
    [stringBuffer appendFormat:@"%02X", (NSUInteger)dataBuffer[i]];
}
NSLog(@"0-8 bytes HEXADECIMAL: %@",stringBuffer);

如您所见,0x3700000000000000 是接下来的 8 个字节。为了访问接下来的 8 个字节,我必须做的唯一更改是将 SeekFileToOffset 的值更改为 8,以便访问接下来的 8 个字节的数据。

总而言之,您给我的解决方案很有用,但是手动输入十六进制值是不切实际的。如果将字节格式化为字符串然后解析它们不是这样做的方法,那么我如何直接访问数据的前 8 个字节并将它们转换为字节指针?

4

2 回答 2

3

4,294,967,295 是ULONG_MAX,所以看起来扫描仪只是溢出了。(该scanHexInt方法会吃掉它找到的所有六进制数字,并且由于零也是一个六进制数字,因此该方法会吞掉整个数字,越过ULONG_MAX.)

于 2012-11-21T09:06:41.873 回答
3

您似乎正在尝试将存储在 an 中的 8 个字节转换NSData为 64 位整数(0x3700000000000000 中有 8 个字节)。

如果这是您的目标,那么通过将字节格式化为字符串然后对其进行解析不是这样做的方法。您需要找出数据的字节序格式(即最低或最高有效字节优先),然后使用适当的字节序转换函数。

顺便说一句,您获得 4294967295 (0xFFFFFFFF) 的原因是因为您要求NSScanner读取 32 位整数并且您的数字 (0x3700000000000000) 溢出。

对评论的回应

由于 x86 是 little-endian,您可以通过强制转换直接从字节指针读取 little-endian 64 位整数:

Byte bytes[] = { 0x37, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 };
NSLog(@"%lld", *(int64_t *)bytes);

如果您需要读取 big-endian 数据,那么您可以使用 endian 转换功能,例如Endian64_Swap- 它只是翻转您拥有的数据,所以读取然后翻转。

请注意,x86 不需要对齐数据访问,但如果对齐,性能会更好。

于 2012-11-21T09:35:35.573 回答