私はWindowsサーバからUTF-8でエンコードされたNSData
を持っています、そしてそれをiPhone用のNSString
に変換したいです。データには両方のプラットフォームで値が異なる文字(度数記号など)が含まれているため、 データを文字列に変換するにはどうすればよいですか。
データがnullで終了していない場合は、-initWithData:encoding:
を使用してください。
NSString* newStr = [[NSString alloc] initWithData:theData encoding:NSUTF8StringEncoding];
データがnullで終了する場合は、最後に余分な-stringWithUTF8String:
を使用しないように、代わりに\0
を使用してください。
NSString* newStr = [NSString stringWithUTF8String:[theData bytes]];
(入力が正しくUTF-8エンコードされていない場合は、nil
が返されます。)
let newStr = String(data: data, encoding: .utf8)
// note that `newStr` is a `String?`, not a `String`.
データがヌル文字で終了している場合は、そのヌル文字を削除する安全な方法、または上記のObjective-Cバージョンと同様の安全でない方法を使用することができます。
// safe way, provided data is \0-terminated
let newStr1 = String(data: data.subdata(in: 0 ..< data.count - 1), encoding: .utf8)
// unsafe way, provided data is \0-terminated
let newStr2 = data.withUnsafeBytes(String.init(utf8String:))
あなたはこのメソッドを呼び出すことができます
+(id)stringWithUTF8String:(const char *)bytes.
私はこれを煩わしくしないように謙虚にカテゴリを提出します。
@interface NSData (EasyUTF8)
// Safely decode the bytes into a UTF8 string
- (NSString *)asUTF8String;
@end
そして
@implementation NSData (EasyUTF8)
- (NSString *)asUTF8String {
return [[NSString alloc] initWithData:self encoding:NSUTF8StringEncoding];
}
@end
(ARCを使用していない場合は、そこにautorelease
が必要です。)
今は明らかに冗長ではなく:
NSData *data = ...
[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
できるよ:
NSData *data = ...
[data asUTF8String];
文字列からデータへそして文字列へ戻るスウィフトバージョン:
Xcode 10.1•Swift 4.2.1
extension Data {
var string: String? {
return String(data: self, encoding: .utf8)
}
}
extension StringProtocol {
var data: Data {
return Data(utf8)
}
}
extension String {
var base64Decoded: Data? {
return Data(base64Encoded: self)
}
}
遊び場
let string = "Hello World" // "Hello World"
let stringData = string.data // 11 bytes
let base64EncodedString = stringData.base64EncodedString() // "SGVsbG8gV29ybGQ="
let stringFromData = stringData.string // "Hello World"
let base64String = "SGVsbG8gV29ybGQ="
if let data = base64String.base64Decoded {
print(data) // 11 bytes
print(data.base64EncodedString()) // "SGVsbG8gV29ybGQ="
print(data.string ?? "nil") // "Hello World"
}
let stringWithAccent = "Olá Mundo" // "Olá Mundo"
print(stringWithAccent.count) // "9"
let stringWithAccentData = stringWithAccent.data // "10 bytes" note: an extra byte for the acute accent
let stringWithAccentFromData = stringWithAccentData.string // "Olá Mundo\n"
時々、他の答えの方法はうまくいかない。私の場合は、RSA秘密鍵を使って署名を生成しています。結果はNSDataです。私はこれがうまくいくように思われることがわかりました:
Objective-C
NSData *signature;
NSString *signatureString = [signature base64EncodedStringWithOptions:0];
スイフト
let signatureString = signature.base64EncodedStringWithOptions(nil)
要約すると、完全な答えがここにあります。
私の問題は私が使ったときに
[NSString stringWithUTF8String:(char *)data.bytes];
私が得た文字列は予測できませんでした。約70%には期待値が含まれていましたが、あまりにも多くの場合Null
またはさらに悪い結果となりました。
掘った後、私は
[[NSString alloc] initWithBytes:(char *)data.bytes length:data.length encoding:NSUTF8StringEncoding];
毎回期待通りの結果が得られました。
Swift 4.2では、UTF-8を使用してString
インスタンスをData
インスタンスに変換するためにString
の init(data:encoding:)
initializerを使用できます。 init(data:encoding:)
は次のように宣言されています。
init?(data: Data, encoding: String.Encoding)
与えられたエンコーディングを使って与えられたデータをUnicode文字に変換することによって初期化された
String
を返します。
次のプレイグラウンドコードは、その使い方を示しています。
import Foundation
let json = """
{
"firstName" : "John",
"lastName" : "Doe"
}
"""
let data = json.data(using: String.Encoding.utf8)!
let optionalString = String(data: data, encoding: String.Encoding.utf8)
print(String(describing: optionalString))
/*
prints:
Optional("{\n\"firstName\" : \"John\",\n\"lastName\" : \"Doe\"\n}")
*/