แปลง NSData ที่เข้ารหัส UTF-8 เป็น NSString


567

ฉันเข้ารหัส UTF-8 NSDataจากเซิร์ฟเวอร์ windows และฉันต้องการแปลงเป็นNSStringiPhone เนื่องจากข้อมูลมีอักขระ (เช่นสัญลักษณ์องศา) ซึ่งมีค่าแตกต่างกันในทั้งสองแพลตฟอร์มฉันจะแปลงข้อมูลเป็นสตริงได้อย่างไร


16
UTF-8 คือ UTF-8 ทุกที่ เมื่อเป็น UTF-8 จะไม่มีค่าแตกต่างกันสำหรับแพลตฟอร์มที่แตกต่างกัน นั่นคือจุดรวมของมัน
gnasher729

คำตอบ:


1155

หากข้อมูลไม่ถูกยกเลิกเป็นโมฆะคุณควรใช้ -initWithData:encoding:

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

หากข้อมูลถูกยกเลิกเป็นโมฆะคุณควรใช้-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 นั้นหรือวิธีที่ไม่ปลอดภัยคล้ายกับเวอร์ชัน 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 ไม่ผ่านมันอาร์กิวเมนต์โมฆะหรือมันจะโยนยกเว้น
JasonZ

31
MIND สิ่งนี้: เมื่อใช้ "stringWithUTF8String:" บนสตริงที่ไม่สิ้นสุดด้วยค่า null ผลลัพธ์จะคาดเดาไม่ได้!
Berik

2
โซลูชันทั้งสองคืนกลับมาให้ฉัน
Husyn

1
คุณจะรู้ได้อย่างไรว่า NSData ของคุณสิ้นสุดด้วย null หรือไม่ ดูคำตอบที่ทอมแฮร์ริงที่: stackoverflow.com/questions/27935054/... จากประสบการณ์ของฉันเราไม่ควรคิดว่า NSData จะถูกยกเลิกด้วย null หรือไม่: มันอาจแตกต่างจากการส่งสัญญาณหนึ่งไปยังการส่งต่อไปแม้กระทั่งจากเซิร์ฟเวอร์ที่รู้จัก
Elise van Looij

1
@ElisevanLooij ขอบคุณสำหรับลิงค์ ฉันยืนยันว่าหากข้อมูลที่ส่งสามารถสุ่มสิ้นสุดด้วยค่า null หรือไม่โปรโตคอลไม่ถูกกำหนด
kennytm

28

คุณสามารถเรียกวิธีนี้ได้

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

27
เฉพาะในกรณีที่ข้อมูลถูกยกเลิกเป็นโมฆะ ซึ่งมันอาจจะไม่ใช่ (และอันที่จริงแล้วอาจไม่ใช่)
Ivan Vučica

ผมไม่ทราบว่าทำไมบนโลกนี้จะแตกในสายที่ไม่สิ้นสุดด้วย null เห็นวิธีการที่NSDataรู้ว่าหลายไบต์มันมี ...
Claudiu

5
@Claudiu คุณไม่ได้ผ่านวัตถุ NSData คุณจะได้รับ (const char *) ที่ได้รับด้วย [data bytes] ซึ่งเป็นเพียงตัวชี้ไม่มีข้อมูลขนาด ดังนั้นบล็อกข้อมูลที่ชี้ไปจะต้องถูกยกเลิก ตรวจสอบเอกสารมันบอกอย่างชัดเจน
jbat100

1
@ jbat100: แน่นอน ฉันไม่ชัดเจน ฉันหมายถึงเนื่องจากเป็นไปได้ที่จะเปลี่ยนจากการไม่สิ้นสุด - NSDataเป็นNSString(ดูคำตอบของ KennyTM) ฉันประหลาดใจที่มันไม่ได้+(id)stringWithUTF8Data:(NSData *)dataผล
Claudiu

stringWithUTF8Data ดังนั้นพวกเราส่วนใหญ่จึงสร้างหมวดหมู่ NSString + Foo และสร้างวิธีการ
วิลเลียม 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นั่น)

ตอนนี้แทน verbose ที่น่ากลัว:

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

คุณทำได้:

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

18

เวอร์ชั่น Swift จาก String ไปยัง Data และกลับสู่ String:

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

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

1

เพื่อสรุปนี่เป็นคำตอบที่สมบูรณ์สำหรับฉัน

ปัญหาของฉันคือเมื่อฉันใช้

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

สตริงที่ฉันได้รับคาดเดาไม่ได้: ประมาณ 70% มันมีค่าที่คาดไว้ แต่บ่อยครั้งที่ผลลัพธ์นั้นมีNullหรือแย่กว่านั้นคือ garbaged ที่ส่วนท้ายของสตริง

หลังจากขุดฉันเปลี่ยนไป

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

และได้ผลตามที่คาดหวังทุกครั้ง


เป็นสิ่งสำคัญที่คุณจะต้องเข้าใจ <i> ทำไม </i> คุณถึงได้ผลลัพธ์เป็น 'ขยะ'
Edgar Aroutiounian

1

กับสวิฟท์ 5 คุณสามารถใช้String's init(data:encoding:)initializer เพื่อแปลงDataตัวอย่างเป็นStringเช่นใช้ UTF-8 init(data:encoding:)มีการประกาศดังต่อไปนี้:

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

ส่งคืนStringค่าเริ่มต้นโดยการแปลงข้อมูลที่ให้เป็นอักขระ Unicode โดยใช้การเข้ารหัสที่กำหนด

รหัสสนามเด็กเล่นต่อไปนี้แสดงวิธีใช้:

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.