NSData
WindowsサーバーからエンコードされたUTF-8を持っているので、それをNSString
iPhone に変換したいと思います。データには、両方のプラットフォームで異なる値を持つ文字(度記号など)が含まれているため、データを文字列に変換するにはどうすればよいですか?
NSData
WindowsサーバーからエンコードされたUTF-8を持っているので、それをNSString
iPhone に変換したいと思います。データには、両方のプラットフォームで異なる値を持つ文字(度記号など)が含まれているため、データを文字列に変換するにはどうすればよいですか?
回答:
データが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`.
データがnullで終了している場合は、null文字を削除する安全な方法、または上記の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.
NSData
、それは...持っているどのように多くのバイトを知っています
NSData
していないものからNSString
(KennyTMの回答を参照)に移行することが可能であることを考えると、うまく機能する+(id)stringWithUTF8Data:(NSData *)data
ものがないことに驚いています。
これを煩わしくないようにするために、私はカテゴリを謙虚に提出します。
@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];
文字列からデータへ、そして文字列へ戻るSwiftバージョン:
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)
[[NSData alloc] initWithBase64EncodedString:signatureString options:0]
。 スウィフト: NSData(base64EncodedString: str options: nil)
要約すると、ここで私のために働いた完全な答えがあります。
私の問題は、私が使用したとき
[NSString stringWithUTF8String:(char *)data.bytes];
取得した文字列は予測できませんでした。約70%に期待値が含まれていましたが、次のような結果になりました。 Null
ましたが、文字列の最後に文字化けがたり、さらに悪化たりしました。
掘り下げた後、私はに切り替えました
[[NSString alloc] initWithBytes:(char *)data.bytes length:data.length encoding:NSUTF8StringEncoding];
そして毎回期待通りの結果を得ました。
Swift 5では、UTF-8を使用してインスタンスをインスタンスに変換するために、String
のinit(data:encoding:)
初期化子を使用できます。次の宣言があります:Data
String
init(data:encoding:)
init?(data: Data, encoding: String.Encoding)
String
指定されたエンコーディングを使用して、指定されたデータをUnicode文字に変換することによって初期化されたを返します。
次のPlaygroundコードは、その使用方法を示しています。
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}")
*/