ฉันเข้ารหัส UTF-8 NSData
จากเซิร์ฟเวอร์ windows และฉันต้องการแปลงเป็นNSString
iPhone เนื่องจากข้อมูลมีอักขระ (เช่นสัญลักษณ์องศา) ซึ่งมีค่าแตกต่างกันในทั้งสองแพลตฟอร์มฉันจะแปลงข้อมูลเป็นสตริงได้อย่างไร
ฉันเข้ารหัส UTF-8 NSData
จากเซิร์ฟเวอร์ windows และฉันต้องการแปลงเป็นNSString
iPhone เนื่องจากข้อมูลมีอักขระ (เช่นสัญลักษณ์องศา) ซึ่งมีค่าแตกต่างกันในทั้งสองแพลตฟอร์มฉันจะแปลงข้อมูลเป็นสตริงได้อย่างไร
คำตอบ:
หากข้อมูลไม่ถูกยกเลิกเป็นโมฆะคุณควรใช้ -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:))
คุณสามารถเรียกวิธีนี้ได้
+(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
นั่น)
ตอนนี้แทน verbose ที่น่ากลัว:
NSData *data = ...
[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
คุณทำได้:
NSData *data = ...
[data asUTF8String];
เวอร์ชั่น 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"
บางครั้งวิธีการในคำตอบอื่น ๆ ไม่ทำงาน ในกรณีของฉันฉันกำลังสร้างลายเซ็นด้วยคีย์ส่วนตัวของฉัน RSA และผลลัพธ์คือ NSData ฉันพบว่าสิ่งนี้ดูเหมือนจะใช้ได้:
Objective-C
NSData *signature;
NSString *signatureString = [signature base64EncodedStringWithOptions:0];
รวดเร็ว
let signatureString = signature.base64EncodedStringWithOptions(nil)
[[NSData alloc] initWithBase64EncodedString:signatureString options:0]
; Swift : NSData(base64EncodedString: str options: nil)
เพื่อสรุปนี่เป็นคำตอบที่สมบูรณ์สำหรับฉัน
ปัญหาของฉันคือเมื่อฉันใช้
[NSString stringWithUTF8String:(char *)data.bytes];
สตริงที่ฉันได้รับคาดเดาไม่ได้: ประมาณ 70% มันมีค่าที่คาดไว้ แต่บ่อยครั้งที่ผลลัพธ์นั้นมีNull
หรือแย่กว่านั้นคือ garbaged ที่ส่วนท้ายของสตริง
หลังจากขุดฉันเปลี่ยนไป
[[NSString alloc] initWithBytes:(char *)data.bytes length:data.length encoding:NSUTF8StringEncoding];
และได้ผลตามที่คาดหวังทุกครั้ง
กับสวิฟท์ 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}")
*/