参考以下问题: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);
输出:
接下来,我读取并偏移文件的前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个字节并将它们转换为字节指针?
答案 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不会要求数据访问对齐,但如果匹配则会更好。