将NSData转换为Hex NSString

Daw*_*son 5 cocoa hex objective-c nsdata nsscanner

参考以下问题:将NSData转换为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]];
}
Run Code Online (Sandbox Code Playgroud)

但是,有一个小问题,如果后面有额外的零,则字符串值会有所不同.例如.如果hexa数据是字符串@"3700000000000000",则使用扫描程序转换为整数时:

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

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

那么如何摆脱零呢?

编辑:(回应CRD)

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

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

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

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

输出:

在此输入图像描述

接下来,我读取并偏移文件的前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);
Run Code Online (Sandbox Code Playgroud)

如您所见,0x3700000000000000是接下来的8个字节.我要访问接下来的8个字节的唯一变化是将SeekFileToOffset的值更改为8,以便访问接下来的8个字节的数据.

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

CRD*_*CRD 3

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

如果这是您的目标,那么通过将字节格式化为字符串然后解析它们来实现它并不是实现此目的的方法。您需要找出数据采用的字节序格式(即首先是最低有效字节或最高有效字节),然后使用适当的字节序转换函数。

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

对评论的回应

由于 x86 是小端字节序,因此您可以通过强制转换直接从字节指针读取小端字节序 64 位整数:

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

如果您需要读取大尾数数据,那么您可以使用尾数转换函数,例如Endian64_Swap- 它只会翻转您所拥有的内容,因此先读取然后翻转。

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