Преобразование NSData в int - PullRequest
19 голосов
/ 07 сентября 2011

У меня есть объект NSData, длина составляет 4 байта. Эти четыре байта я извлекаю из другого объекта NSData, используя,

fourByteData=[completeData subdataWithRange:NSMakeRange(0, 16)];

Мой первый вопрос: даст ли приведенный выше оператор мнепервые четыре байта полных данных.

Если да, то как преобразовать все эти байты в эквивалентное целое число.

Ответы [ 7 ]

32 голосов
/ 07 сентября 2011

Является ли значение 268566528 ожидаемым или 528? Если правильное значение равно 528, то порядок байтов имеет порядок с прямым порядком байтов, но процессор имеет порядок с прямым порядком байтов, байты должны быть обращены.

Итак, если правильное значение должно быть 528, то:

NSData *data4 = [completeData subdataWithRange:NSMakeRange(0, 4)];
int value = CFSwapInt32BigToHost(*(int*)([data4 bytes]));

Также обратите внимание, что стандартным порядком сети является big-endian.

25 голосов
/ 07 сентября 2011

Этот оператор даст вам первые 16 байтов данных, а не 4. Чтобы получить первые 4 байта, вам нужно изменить свой оператор на:

fourByteData = [completeData subdataWithRange:NSMakeRange(0, 4)];

Чтобы прочитать данные из NSData Object в целое число, вы можете сделать что-то вроде:

int theInteger;
[completeData getBytes:&theInteger length:sizeof(theInteger)];

Вам не нужно получать первые 4 байта, если вы просто конвертируете их в целое число. Вы можете сделать это прямо из двух строк выше, и ваш completeData приемник

12 голосов
/ 07 сентября 2011

Нет, вы получите 16 байтов данных, поскольку диапазон от смещения 0, а затем 16 байтов.

Если у вас был экземпляр NSData с 4 байтами, вы могли бы выполнить простое приведение типа, напримерэто:

int value = *(int*)([data bytes]);
5 голосов
/ 28 сентября 2012
- (unsigned)parseIntFromData:(NSData *)data{

    NSString *dataDescription = [data description];
    NSString *dataAsString = [dataDescription substringWithRange:NSMakeRange(1, [dataDescription length]-2)];

    unsigned intData = 0;
    NSScanner *scanner = [NSScanner scannerWithString:dataAsString];
    [scanner scanHexInt:&intData];

    return intData;
}

int numberOfChunks = [self parseIntFromData:data];
4 голосов
/ 29 октября 2013

Целочисленное значение iOS хранит младший байт младшего разряда в первом байте и младший байт в последнем байте.У меня есть процедура преобразования, чтобы проверить все эти вещи.проверьте здесь,

Тест ...

int i = 2342342;
    NSData * d1 = [Util dataFromInt:i]; // {[MSB], ..., ... ,[LSB]} <0023bdc6>
    NSData * d2 = [NSData dataWithBytes:&i length:4];  // {[LSB], ..., ... ,[MSB]} <c6bd2300>
    int ci1 = [Util intFromData:d1];
    int ci2 = [Util intFromDataReverse:d2];

Util.m

+ (NSData *) dataFromInt:(int)num {
    unsigned char * arr = (unsigned char *) malloc(sizeof(num) * sizeof(unsigned char));
    for (int i = sizeof(num) - 1 ; i >= 0; i --) {
        arr[i] = num & 0xFF;
        num = num >> 8;
    }
    NSData * data = [NSData dataWithBytes:arr length:sizeof(num)];
    free(arr);
    return data;
}

// {[MSB], ..., ... ,[LSB]}
+ (int) intFromData:(NSData *)data
{
    int intSize = sizeof(int); // change it to fixe length
    unsigned char * buffer = malloc(intSize * sizeof(unsigned char));
    [data getBytes:buffer length:intSize];
    int num = 0;
    for (int i = 0; i < intSize; i++) {
        num = (num << 8) + buffer[i];
    }
    free(buffer);
    return num;
}

// {[LSB], ..., ... ,[MSB]}
+ (int) intFromDataReverse:(NSData *)data
{
    int intSize = sizeof(int);// change it to fixe length
    unsigned char * buffer = malloc(intSize * sizeof(unsigned char));
    [data getBytes:buffer length:intSize];
    int num = 0;
    for (int i = intSize - 1; i >= 0; i--) {
        num = (num << 8) + buffer[i];
    }
    free(buffer);
    return num;
}
1 голос
/ 05 января 2016

Это зависит от нотации Endianness данных, которые вы хотите преобразовать, относительно нотации Endianness вашего устройства. вики на Endianness

Для простоты вам нужно проверить, есть ли два метода

NSData *data4 = [completeData subdataWithRange:NSMakeRange(0, 4)];
int value = CFSwapInt32BigToHost(*(int*)([data4 bytes]));

или

NSData *data4 = [completeData subdataWithRange:NSMakeRange(0, 4)];
int value = CFSwapInt32LittleToHost(*(int*)([data4 bytes]));

и проверить, какойбольше смысла при разборе данных.

0 голосов
/ 21 августа 2018

Предполагая, что _vertexData здесь NSData, и вы знаете, какие данные (типы) ожидать в буфере, вы можете перебирать этот блок с помощью свойства .length NSData.В этом примере каждый блок данных был 32 байта (для хранения 8 x значений с плавающей запятой ), и меня интересовало ведение журнала, начиная с 5-го значения с плавающей запятой

float a,b,c,d; //prepare some values, no need to initialize

// loop thru assuming 8 floats are stored after each other
for (NSUInteger v = 0; v < _vertexData.length; v += sizeof(float)*8 ) {

    // set a starting point for the range, here the 5th float
    NSUInteger shift = v + (sizeof(float)*4);

    // store result in a..
    [_vertexData getBytes:&a range:NSMakeRange(shift,sizeof(a))];

    // increase the starting point by the size of data before
    shift += sizeof(a);

    [_vertexData getBytes:&b range:NSMakeRange(shift,sizeof(b))];

    shift += sizeof(b);
    [_vertexData getBytes:&c range:NSMakeRange(shift,sizeof(c))];

    shift += sizeof(c);
    [_vertexData getBytes:&d range:NSMakeRange(shift,sizeof(d))];

    fprintf(stderr, "r%f, g%f, b%f, a%f \n", a,b,c,d );
}

это можно было бы написать гораздо короче, но ради ясности и с меньшим использованием пропусков лидирующих мисс

возможно, это кому-то поможет

...