4

我正在我的 iPhone 应用程序中使用网络服务。Web 服务方法返回一个包含多个字段(例如 ID、描述等)的响应。其中一个字段包含我需要在 iPhone 应用程序中转换为 UIImage 的二进制图像数据。

我正在使用 NSXMLParser 成功地从 XML 响应中提取数据。在parser:foundCharacters:XMLParser 的选择器中,它NSString*指向每个字段中的字符串。因为这是一个字符串,所以当我遇到图像字段时,我会这样做来读取图像数据:

UIImage *img = [[UIImage alloc] initWithData:[string dataUsingEncoding:NSUTF8StringEncoding]];

但是在这一行之后,img 变量仍然是“nil”。似乎来自 XML 字符串的数据与转换不兼容。我在这里做错了什么?(我能够将其他字段读取到我的变量中,但不能读取此图像数据字段)

提前致谢..

4

2 回答 2

5

按照 fbrereto 的回答,我设法使用此链接上的代码示例将字符串转换为 NSData:http: //www.cocoadev.com/index.pl? BaseSixtyFour(滚动到 MiloBird 用户的代码示例)。现在我可以成功构建图像了。

它使用 objective-c 类别将选择器添加+ (id)dataWithBase64EncodedString:(NSString *)string;到 NSData 类。这是我从上述链接中提取的必要代码示例:

//MBBase64.h

@interface NSData (MBBase64)

+ (id)dataWithBase64EncodedString:(NSString *)string;     //  Padding '=' characters are optional. Whitespace is ignored.

@end


//MBBase64.m

static const char encodingTable[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";

@implementation NSData (MBBase64)

+ (id)dataWithBase64EncodedString:(NSString *)string;
{
    if (string == nil)
        [NSException raise:NSInvalidArgumentException format:nil];
    if ([string length] == 0)
        return [NSData data];

    static char *decodingTable = NULL;
    if (decodingTable == NULL)
    {
        decodingTable = malloc(256);
        if (decodingTable == NULL)
            return nil;
        memset(decodingTable, CHAR_MAX, 256);
        NSUInteger i;
        for (i = 0; i < 64; i++)
            decodingTable[(short)encodingTable[i]] = i;
    }

    const char *characters = [string cStringUsingEncoding:NSASCIIStringEncoding];
    if (characters == NULL)     //  Not an ASCII string!
        return nil;
    char *bytes = malloc((([string length] + 3) / 4) * 3);
    if (bytes == NULL)
        return nil;
    NSUInteger length = 0;

    NSUInteger i = 0;
    while (YES)
    {
        char buffer[4];
        short bufferLength;
        for (bufferLength = 0; bufferLength < 4; i++)
        {
            if (characters[i] == '\0')
                break;
            if (isspace(characters[i]) || characters[i] == '=')
                continue;
            buffer[bufferLength] = decodingTable[(short)characters[i]];
            if (buffer[bufferLength++] == CHAR_MAX)      //  Illegal character!
            {
                free(bytes);
                return nil;
            }
        }

        if (bufferLength == 0)
            break;
        if (bufferLength == 1)      //  At least two characters are needed to produce one byte!
        {
            free(bytes);
            return nil;
        }

        //  Decode the characters in the buffer to bytes.
        bytes[length++] = (buffer[0] << 2) | (buffer[1] >> 4);
        if (bufferLength > 2)
            bytes[length++] = (buffer[1] << 4) | (buffer[2] >> 2);
        if (bufferLength > 3)
            bytes[length++] = (buffer[2] << 6) | buffer[3];
    }

    realloc(bytes, length);
    return [NSData dataWithBytesNoCopy:bytes length:length];
}

@end

谢谢你们!

于 2010-03-04T05:18:21.123 回答
1

诀窍NSString是存在与其包含的数据相关的隐式编码。但是,您收到的图像可能是无法正确转换的格式,因为它要么是二进制数据,要么是二进制数据的某种无损编码(例如,base64)。因此,诀窍是确保您根本不让NSString执行任何编码转换,否则您的图像数据将被破坏。而不是使用dataUsingEncoding:我会尝试一个更像getBytes:maxLength:usedLength:encoding:options:range:remainingRange. 虽然比dataUsingEncoding:我认为的更复杂,但它会给您提供仅从 NSString 获取数据所需的灵活性,仅此而已。

于 2010-03-02T05:01:17.133 回答