UTF-8 kodlu NSData'yı NSString'e dönüştürme


567

NSDataWindows sunucusundan kodlanmış UTF-8 var ve NSStringiPhone için dönüştürmek istiyorum . Veriler her iki platformda da farklı değerlere sahip karakterler (derece sembolü gibi) içerdiğinden, verileri dizeye nasıl dönüştürebilirim?


16
UTF-8 her yerde UTF-8'dir. UTF-8 olduğunda, farklı platformlar için farklı değerler yoktur. Bütün mesele bu.
gnasher729

Yanıtlar:


1155

Veriler boş değerli değilse, şunu kullanmalısınız: -initWithData:encoding:

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

Veriler null olarak sonlandırılırsa, sonunda -stringWithUTF8String:ekstradan kaçınmak için bunu kullanmalısınız \0.

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

(Giriş düzgün UTF-8 kodlu değilse, alacağınızı unutmayın nil.)


Hızlı varyant:

let newStr = String(data: data, encoding: .utf8)
// note that `newStr` is a `String?`, not a `String`.

Veriler null olarak sonlandırılırsa, o null karakteri kaldıran güvenli yoldan veya yukarıdaki Objective-C sürümüne benzer güvenli olmayan yoldan gidebilirsiniz.

// 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
dikkat et!! stringWithUTF8String kullanıyorsanız, bir NULL argümanı geçirmeyin yoksa bir istisna atar
JasonZ

31
BUNU BU: boş sonlandırılmamış bir dize "stringWithUTF8String:" kullanırken, sonuç tahmin edilemez!
Berik

2
Her iki çözüm de benim için sıfır.
Husyn

1
NSData dosyanızın sonlandırılıp sonlandırılmadığını nasıl anlarsınız? Tom Harrington'un cevabına bakınız: stackoverflow.com/questions/27935054/… . Deneyimlerime göre, NSData'nın null sonlandırıldığını veya sonlandırılmadığını varsaymamalıyız: bilinen bir sunucudan bile bir iletimden diğerine farklılık gösterebilir.
Elise van Looij

1
@ElisevanLooij Bağlantı için teşekkürler. İletilen verilerin rastgele boşaltılabileceğini veya protokolün kötü tanımlanamayacağını iddia ediyorum.
kennytm

28

Bu yöntemi çağırabilirsiniz

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

27
Yalnızca veriler boş bırakılırsa. Hangi olmayabilir (ve aslında, muhtemelen değil).
Ivan Vučica

neden dünyada NSDatabilmiyorum kaç bayt olduğunu bilir görmek null olmayan sonlandırılmış dizeleri kırmak bilmiyorum ...
Claudiu

5
@Claudiu, bir NSData nesnesini geçirmiyorsunuz, [veri baytları] ile elde edilen bir (const char *) iletiyorsunuz, bu sadece bir işaretçi, boyut bilgisi yok. Bu nedenle işaret ettiği veri bloğunun boş bırakılması gerekir. Belgelere göz atın, açıkça söylüyor.
jbat100

1
@ jbat100: Elbette. Net değildim. Demek istediğim, null olmayan NSDatabir NSStringsondan bir (KennyTM'nin cevabına bakın) geçmesi mümkün olduğu göz önüne alındığında, +(id)stringWithUTF8Data:(NSData *)datasadece işe yarayan bir şey olmadığına şaşırdım .
Claudiu

stringWithUTF8Data, dolayısıyla çoğumuz bir NSString + Foo kategorisi oluştururuz ve yöntemi yaratırız.
William Cerniuk

19

Bunu daha az sinir bozucu yapmak için alçakgönüllü bir kategori gönderirim:

@interface NSData (EasyUTF8)

// Safely decode the bytes into a UTF8 string
- (NSString *)asUTF8String;

@end

ve

@implementation NSData (EasyUTF8)

- (NSString *)asUTF8String {
    return [[NSString alloc] initWithData:self encoding:NSUTF8StringEncoding];    
}

@end

(ARC kullanmıyorsanız autoreleaseoraya ihtiyacınız olacaktır .)

Şimdi dehşet verici derecede ayrıntılı:

NSData *data = ...
[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];

Yapabilirsin:

NSData *data = ...
[data asUTF8String];

18

Dize'den Veriye ve tekrar Dize'ye Swift sürümü:

Xcode 10.1 • Hızlı 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)
    }
}

Oyun alanı

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

Bazen, diğer cevaplardaki yöntemler işe yaramaz. Benim durumumda, RSA özel anahtarımla bir imza oluşturuyorum ve sonuç NSData. Bunun işe yaradığını gördüm:

Objective-C

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

hızlı

let signatureString = signature.base64EncodedStringWithOptions(nil)

bu dizeyi nsdata'ya nasıl alabilirim?
Darshan Kunjadiya

1
@DarshanKunjadiya: Amaç-C : [[NSData alloc] initWithBase64EncodedString:signatureString options:0]; Swift : NSData(base64EncodedString: str options: nil)
mikeho

1

Özetlemek gerekirse, işte tam bir cevap, benim için işe yaradı.

Benim sorunum,

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

Aldığım dize tahmin edilemezdi: yaklaşık% 70'i beklenen değeri içeriyordu, ancak çoğu zaman Nullveya daha da kötüsü ile sonuçlandı : ipin sonunda garbaged.

Biraz kazıdan sonra

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

Ve her seferinde beklenen sonucu aldım.


Neden "çöp" sonuçları aldığınızı anlamanız önemlidir.
Edgar Aroutiounian

1

Swift 5 ile UTF-8 kullanarak bir örneği örneğe dönüştürmek için String'ın init(data:encoding:)başlatıcısını kullanabilirsiniz . aşağıdaki beyanı içerir:DataStringinit(data:encoding:)

init?(data: Data, encoding: String.Encoding)

StringBelirli bir kodlamayı kullanarak belirli verileri Unicode karakterlere dönüştürerek başlatılmış değeri döndürür .

Aşağıdaki Oyun Alanı kodu nasıl kullanılacağını gösterir:

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}")
*/
Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.