如何转换NSData
为base64
. 我有NSData
并且想要转换成base64
我该怎么做?
7 回答
编辑
从 OS X 10.9 / iOS 7 开始,这是内置在框架中的。
看-[NSData base64EncodedDataWithOptions:]
在 iOS7/OS X 10.9 之前:
Matt Gallagher 写了一篇关于这个主题的文章。在底部,他提供了指向他的 iPhone 可嵌入代码的链接。
在 Mac 上你可以使用 OpenSSL 库,在 iPhone 上他编写了自己的 impl。
//from: http://cocoadev.com/BaseSixtyFour
+ (NSString*)base64forData:(NSData*)theData {
const uint8_t* input = (const uint8_t*)[theData bytes];
NSInteger length = [theData length];
static char table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
NSMutableData* data = [NSMutableData dataWithLength:((length + 2) / 3) * 4];
uint8_t* output = (uint8_t*)data.mutableBytes;
NSInteger i;
for (i=0; i < length; i += 3) {
NSInteger value = 0;
NSInteger j;
for (j = i; j < (i + 3); j++) {
value <<= 8;
if (j < length) {
value |= (0xFF & input[j]);
}
}
NSInteger theIndex = (i / 3) * 4;
output[theIndex + 0] = table[(value >> 18) & 0x3F];
output[theIndex + 1] = table[(value >> 12) & 0x3F];
output[theIndex + 2] = (i + 1) < length ? table[(value >> 6) & 0x3F] : '=';
output[theIndex + 3] = (i + 2) < length ? table[(value >> 0) & 0x3F] : '=';
}
return [[[NSString alloc] initWithData:data encoding:NSASCIIStringEncoding] autorelease];
}
作为更新,iOS7 SDK 有一个关于NSData (NSDataBase64Encoding)
with 方法的类别
-[NSData base64EncodedStringWithOptions:]
-[NSData initWithBase64EncodedString:options:]
-[NSData initWithBase64EncodedData:options:]
-[NSData base64EncodedDataWithOptions:]
应该避免滚动你自己的类别方法
超级简单的谷歌库代码在这里。
只需使用+rfc4648Base64StringEncoding
获取实例,然后使用encode
/decode
函数。
这是一件美好的事情。(不过,不要忘记GTMDefines.h
从根目录中获取头文件和头文件。)
这并不容易。因为在 c 或 obj-c 中没有对此的内置支持。这就是我正在做的事情(基本上是让 CL 为我做这件事):
- (NSString *)_base64Encoding:(NSString *) str
{
NSTask *task = [[[NSTask alloc] init] autorelease];
NSPipe *inPipe = [NSPipe pipe], *outPipe = [NSPipe pipe];
NSFileHandle *inHandle = [inPipe fileHandleForWriting], *outHandle = [outPipe fileHandleForReading];
NSData *outData = nil;
[task setLaunchPath:@"/usr/bin/openssl"];
[task setArguments:[NSArray arrayWithObjects:@"base64", @"-e", nil]];
[task setStandardInput:inPipe];
[task setStandardOutput:outPipe];
[task setStandardError:outPipe];
[task launch];
[inHandle writeData:[str dataUsingEncoding: NSASCIIStringEncoding]];
[inHandle closeFile];
[task waitUntilExit];
outData = [outHandle readDataToEndOfFile];
if (outData)
{
NSString *base64 = [[[NSString alloc] initWithData:outData encoding:NSUTF8StringEncoding] autorelease];
if (base64)
return base64;
}
return nil;
}
你像这样使用它:
NSString *b64str = [strToConvert _base64Encoding:strToConvert];
这不是我的代码 - 我在这里找到了它:http: //www.cocoadev.com/index.pl?BaseSixtyFour并且效果很好。你总是可以把它变成一个 +() 方法。
哦,为了让你的 NSData 到一个 NSString 这个方法:
NSString *str = [[NSString alloc] initWithData:data encoding:NSASCIIStringEncoding];
iOS 始终包含对 base64 编码和解码的内置支持。如果您看一下,resolv.h
您应该会看到两个函数b64_ntop
和b64_pton
. Square SocketRocket库提供了一个合理的示例,说明如何使用 Objective-c 中的这些函数。
这些功能经过了很好的测试和可靠 - 与您在随机互联网帖子中可能找到的许多实现不同。不要忘记链接到libresolv.dylib
.
如果您链接到 iOS 7 SDK,您可以使用较新的方法initWithBase64Encoding:
和base64EncodedDataWithOptions:
. 这些存在于以前的版本中,但是是私有的。因此,如果您链接到 6 SDK,您可能会遇到未定义的行为。这将是仅在链接 7 SDK 时如何使用它的示例:
#ifndef __IPHONE_7_0
// oh no! you are using something unsupported!
// Call and implementation that uses b64_pton here
#else
data = [[NSData alloc] initWithBase64Encoding:string];
#endif
我修改了上面的代码以满足我的需要,构建了一个 HTTP POST。我能够跳过 NSString 步骤,并在 BASE64 代码中包含换行符,至少有一个 Web 服务器发现它更可口:
#define LINE_SIZE 76
//originally from: http://www.cocoadev.com/index.pl?BaseSixtyFour
// via joshrl on stockoverflow
- (void) appendBase64Of: (NSData *)inData to:(NSMutableData *)outData {
const uint8_t* input = (const uint8_t*)[inData bytes];
NSInteger length = [inData length];
static char table[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
uint8_t buf[LINE_SIZE + 4 + 2];
size_t n = 0;
NSInteger i;
for (i=0; i < length; i += 3) {
NSInteger value = 0;
NSInteger j;
for (j = i; j < (i + 3); j++) {
value <<= 8;
if (j < length) {
value |= (0xFF & input[j]);
}
}
buf[n + 0] = table[(value >> 18) & 0x3F];
buf[n + 1] = table[(value >> 12) & 0x3F];
buf[n + 2] = (i + 1) < length ? table[(value >> 6) & 0x3F] : '=';
buf[n + 3] = (i + 2) < length ? table[(value >> 0) & 0x3F] : '=';
n += 4;
if (n + 2 >= LINE_SIZE) {
buf[n++] = '\r';
buf[n++] = '\n';
[outData appendBytes:buf length:n];
n = 0;
}
}
if (n > 0) {
buf[n++] = '\r';
buf[n++] = '\n';
[outData appendBytes:buf length:n];
}
return;
}