NSData
をbase64
に変換する方法。 NSData
があり、base64
に変換したいのですが、どうすればよいですか?
[〜#〜] edit [〜#〜]
OS X 10.9/iOS 7現在、これはフレームワークに組み込まれています。
見る -[NSData base64EncodedDataWithOptions:]
IOS7/OS X 10.9より前:
マット・ギャラガーは、まさにこのトピックについて 記事 を書きました。一番下には、iPhone用の埋め込み可能なコードへのリンクがあります。
MacではOpenSSLライブラリを使用でき、iPhoneでは彼が独自の実装を記述します。
//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)
にメソッドを持つカテゴリがあります
-[NSData base64EncodedStringWithOptions:]
-[NSData initWithBase64EncodedString:options:]
-[NSData initWithBase64EncodedData:options:]
-[NSData base64EncodedDataWithOptions:]
独自のカテゴリメソッドをロールする必要はありません
非常に簡単なGoogleライブラリコードのドロップイン こちら
+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
を見ると、2つの関数b64_ntop
とb64_pton
が表示されているはずです。 Square SocketRocket ライブラリは、objective-cからこれらの関数を使用する方法の合理的な例を提供します。
これらの機能は、十分にテストされており、信頼性があります-ランダムなインターネット投稿で見られる多くの実装とは異なります。 libresolv.dylib
に対してリンクすることを忘れないでください。
IOS 7 SDKに対してlinkする場合、新しいメソッド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コードに改行を含めることができました。少なくとも1つの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;
}