我正在iPhone应用中使用网络服务。 Web服务方法返回一个响应,该响应具有多个字段(例如ID,描述等)。其中一个字段包含我需要在我的iPhone应用程序中转换为UIImage的二进制图像数据。
我正在成功使用NSXMLParser从XML响应中提取数据。在XMLParser的parser:foundCharacters:
选择器中,它提供了一个NSString*
指向每个字段中的字符串。由于这是一个字符串,这是我在遇到图像字段时读取图像数据的方法:
UIImage *img = [[UIImage alloc] initWithData:[string dataUsingEncoding:NSUTF8StringEncoding]];
但是img变量在此行之后仍然是“nil”。似乎XML字符串中的数据与转换不兼容。我在这做错了什么? (我能够将其他字段读入我的变量但不能读取此图像数据字段)
提前致谢..
答案 0 :(得分: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
谢谢大家!
答案 1 :(得分:1)
NSString
的技巧是存在与其包含的数据相关联的隐式编码。但是,您接收的图像可能采用无法正确转换的格式,因为它是二进制数据或二进制数据的某些无损编码(例如,base64)。因此,诀窍是确保您不让NSString
执行任何编码转换,否则您的图像数据将被破坏。我没有使用dataUsingEncoding:
,而是尝试更像getBytes:maxLength:usedLength:encoding:options:range:remainingRange
的API。虽然比dataUsingEncoding:
更复杂,但我认为它将为您提供从NSString中获取数据所需的灵活性,仅此而已。