Ho UTF-8 codificato NSData
da Windows Server e voglio convertirlo in NSString
per iPhone. Poiché i dati contengono caratteri (come un simbolo di grado) che hanno valori diversi su entrambe le piattaforme, come posso convertire i dati in stringa?
Ho UTF-8 codificato NSData
da Windows Server e voglio convertirlo in NSString
per iPhone. Poiché i dati contengono caratteri (come un simbolo di grado) che hanno valori diversi su entrambe le piattaforme, come posso convertire i dati in stringa?
Risposte:
Se i dati non sono null, è necessario utilizzare -initWithData:encoding:
NSString* newStr = [[NSString alloc] initWithData:theData encoding:NSUTF8StringEncoding];
Se i dati sono nulli, è necessario utilizzare invece -stringWithUTF8String:
per evitare il extra \0
alla fine.
NSString* newStr = [NSString stringWithUTF8String:[theData bytes]];
(Nota che se l'ingresso non è correttamente codificato UTF-8, otterrai nil
.)
let newStr = String(data: data, encoding: .utf8)
// note that `newStr` is a `String?`, not a `String`.
Se i dati sono terminati con null, è possibile procedere nel modo sicuro che è rimuovere quel carattere null o nel modo non sicuro simile alla versione Objective-C sopra.
// 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:))
Potresti chiamare questo metodo
+(id)stringWithUTF8String:(const char *)bytes.
NSData
sa quanti byte ha ...
NSData
a un NSString
(vedi la risposta di KennyTM), sono sorpreso che non ce ne sia uno +(id)stringWithUTF8Data:(NSData *)data
che funzioni .
Umilmente invio una categoria per rendere questo meno fastidioso:
@interface NSData (EasyUTF8)
// Safely decode the bytes into a UTF8 string
- (NSString *)asUTF8String;
@end
e
@implementation NSData (EasyUTF8)
- (NSString *)asUTF8String {
return [[NSString alloc] initWithData:self encoding:NSUTF8StringEncoding];
}
@end
(Nota che se non usi ARC avrai bisogno di un autorelease
lì.)
Ora invece del terribilmente prolisso:
NSData *data = ...
[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
Tu puoi fare:
NSData *data = ...
[data asUTF8String];
La versione Swift da String a Data e di nuovo a 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)
}
}
Terreno di gioco
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"
A volte, i metodi nelle altre risposte non funzionano. Nel mio caso, sto generando una firma con la mia chiave privata RSA e il risultato è NSData. Ho scoperto che questo sembra funzionare:
Objective-C
NSData *signature;
NSString *signatureString = [signature base64EncodedStringWithOptions:0];
veloce
let signatureString = signature.base64EncodedStringWithOptions(nil)
[[NSData alloc] initWithBase64EncodedString:signatureString options:0]
; Swift : NSData(base64EncodedString: str options: nil)
Per riassumere, ecco una risposta completa, che ha funzionato per me.
Il mio problema era che quando l'ho usato
[NSString stringWithUTF8String:(char *)data.bytes];
La stringa che ho ricevuto era imprevedibile: circa il 70% conteneva il valore atteso, ma troppo spesso risultava con Null
o anche peggio: confonduto alla fine della stringa.
Dopo alcuni scavi sono passato a
[[NSString alloc] initWithBytes:(char *)data.bytes length:data.length encoding:NSUTF8StringEncoding];
E ho ottenuto il risultato atteso ogni volta.
Con Swift 5, puoi utilizzare String
l' init(data:encoding:)
inizializzatore per convertire Data
un'istanza in String
un'istanza utilizzando UTF-8. init(data:encoding:)
ha la seguente dichiarazione:
init?(data: Data, encoding: String.Encoding)
Restituisce un
String
inizializzato convertendo i dati dati in caratteri Unicode utilizzando una determinata codifica.
Il seguente codice Playground mostra come usarlo:
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}")
*/