将NSData转换为Hex NSString

时间:2012-11-21 08:43:02

标签: objective-c cocoa hex nsdata nsscanner

参考以下问题: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]];
}

然而,有一个小问题,如果后面有额外的零,字符串值会有所不同。例如。如果hexa数据是字符串@“3700000000000000”,则使用扫描器转换为整数时:

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

结果将是4294967295,这是不正确的。不应该是55,因为只有hexa 37被采取?

那么如何摆脱零呢?

编辑:(回应CRD)

嗨,谢谢你澄清我的怀疑。那你要做的是直接从一个字节指针读取64位整数吧?不过我有另一个问题。你如何实际将NSData转换为字节指针?

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

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

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

输出:

enter image description here

接下来,我读取并偏移文件的前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个字节并将它们转换为字节指针?

2 个答案:

答案 0 :(得分:3)

4,294,967,295是ULONG_MAX,因此看起来扫描仪只是溢出。 (scanHexInt方法会占用它找到的所有六进制数字,并且由于零也是六位数,因此该方法会覆盖整个数字,超过ULONG_MAX。)

答案 1 :(得分:3)

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

如果这是你的目标,那么通过将字节格式化为字符串然后解析它们来实现它是的方式。您需要找出数据所在的 endian 格式(即最少或最重要的字节)然后使用适当的字节序转换函数。

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

对评论的回应

由于x86是little-endian,你可以直接从字节指针中读取一个小端64位整数:

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

如果您需要阅读大端数据,那么您可以使用诸如Endian64_Swap之类的字节序转换功能 - 它只是翻转您拥有的内容,因此请阅读然后翻转。

请注意,x86不会要求数据访问对齐,但如果匹配则会更好。