web-dev-qa-db-ja.com

UTF-8エンコードのNSDataをNSStringに変換する

私はWindowsサーバからUTF-8でエンコードされたNSDataを持っています、そしてそれをiPhone用のNSStringに変換したいです。データには両方のプラットフォームで値が異なる文字(度数記号など)が含まれているため、 データを文字列に変換するにはどうすればよいですか。

555
Ashwini

データが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:))
1137
kennytm

あなたはこのメソッドを呼び出すことができます

+(id)stringWithUTF8String:(const char *)bytes.
28
Gouldsc

私はこれを煩わしくしないように謙虚にカテゴリを提出します。

@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];
19
Claudiu

文字列からデータへそして文字列へ戻るスウィフトバージョン:

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"
18
Leo Dabus

時々、他の答えの方法はうまくいかない。私の場合は、RSA秘密鍵を使って署名を生成しています。結果はNSDataです。私はこれがうまくいくように思われることがわかりました:

Objective-C

NSData *signature;
NSString *signatureString = [signature base64EncodedStringWithOptions:0];

スイフト

let signatureString = signature.base64EncodedStringWithOptions(nil)
15
mikeho

要約すると、完全な答えがここにあります。

私の問題は私が使ったときに

[NSString stringWithUTF8String:(char *)data.bytes];

私が得た文字列は予測できませんでした。約70%には期待値が含まれていましたが、あまりにも多くの場合Nullまたはさらに悪い結果となりました。

掘った後、私は

[[NSString alloc] initWithBytes:(char *)data.bytes length:data.length encoding:NSUTF8StringEncoding];

毎回期待通りの結果が得られました。

1
Gal

Swift 4.2では、UTF-8を使用してStringインスタンスをDataインスタンスに変換するためにStringinit(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}")
*/
0
Imanou Petit