UTF-8でエンコードされたNSDataをNSStringに変換します


567

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


16
UTF-8はどこでもUTF-8です。UTF-8になると、プラットフォームごとに異なる値はありません。それはそれの全体のポイントです。
gnasher729 2014

回答:


1155

データがnullで終了していない場合は、 -initWithData:encoding:

NSString* newStr = [[NSString alloc] initWithData:theData encoding:NSUTF8StringEncoding];

データがnullで終了している場合は、代わりにを使用-stringWithUTF8String:\0て、最後の余分な部分を回避する必要があります。

NSString* newStr = [NSString stringWithUTF8String:[theData bytes]];

(入力が適切にUTF-8でエンコードされていない場合は、になることに注意してくださいnil。)


Swiftバリアント:

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:))

5
気を付けて!!stringWithUTF8Stringを使用する場合は、NULL引数を渡さないでください。渡すと、例外がスローされます
JasonZ

31
これを覚えておいてください:nullで終了していない文字列に "stringWithUTF8String:"を使用すると、結果は予測できません!
Berik 2012

2
どちらのソリューションでも、nilが返されます。
Husyn

1
NSDataがnullで終了しているかどうかはどのようにしてわかりますか?stackoverflow.com/questions/27935054/…でトムハリントンの回答を参照してください。私の経験では、NSDataがnullで終了しているかどうかを仮定するべきではありません。既知のサーバーからの送信でも、送信ごとに異なる場合があります。
Elise van Looij

1
@ElisevanLooijリンクをありがとう。送信されたデータがランダムにnullで終了するか、プロトコルが正しく定義されていない可能性があると私は主張します。
kennytm

28

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

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

27
データがnullで終了する場合のみ。それはそうではないかもしれません(そして実際、おそらくそうではありません)。
IvanVučica2013年

地球上で、これはどのように見て非nullで終わる文字列に壊れるなぜ私は知らないNSData、それは...持っているどのように多くのバイトを知っています
クラウディウ

5
@ Claudiu、NSDataオブジェクトではなく、[データバイト]で取得された(const char *)を渡しています。これは単なるポインタであり、サイズ情報はありません。したがって、それが指すデータブロックはnullで終了する必要があります。ドキュメントを確認してください、それは明示的に言っています。
jbat100 2013年

1
@ jbat100:もちろんです。はっきりしませんでした。私は、nullで終了NSDataしていないものからNSString(KennyTMの回答を参照)に移行することが可能であることを考えると、うまく機能する+(id)stringWithUTF8Data:(NSData *)dataものがないことに驚いています。
Claudiu 2013年

stringWithUTF8Data。したがって、ほとんどの場合、NSString + Fooカテゴリを作成し、メソッドを作成します。
William Cerniuk

19

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

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

18

文字列からデータへ、そして文字列へ戻る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"

16

他の回答の方法が機能しない場合があります。私の場合、RSA秘密鍵を使用して署名を生成しています。結果はNSDataです。これはうまくいくようです:

Objective-C

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

迅速

let signatureString = signature.base64EncodedStringWithOptions(nil)

その文字列をnsdataに取得する方法
Darshan Kunjadiya、2015

1
@DarshanKunjadiya:のObjective-C :[[NSData alloc] initWithBase64EncodedString:signatureString options:0]スウィフトNSData(base64EncodedString: str options: nil)
mikeho

1

要約すると、ここで私のために働いた完全な答えがあります。

私の問題は、私が使用したとき

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

取得した文字列は予測できませんでした。約70%に期待値が含まれていましたが、次のような結果になりました。 Nullましたが、文字列の最後に文字化けがたり、さらに悪化たりしました。

掘り下げた後、私はに切り替えました

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

そして毎回期待通りの結果を得ました。


あなたが「ゴミ」の結果を得た理由を理解することが重要です。
Edgar Aroutiounian

1

Swift 5では、UTF-8を使用してインスタンスをインスタンスに変換するために、Stringinit(data:encoding:)初期化子を使用できます。次の宣言があります:DataStringinit(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}")
*/
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.