UTF-8로 인코딩 된 NSData를 NSString으로 변환


567

NSDataWindows 서버에서 UTF-8로 인코딩 되어 NSString있으며 iPhone 으로 변환하고 싶습니다 . 데이터에는 두 플랫폼에서 서로 다른 값을 가진 문자 (도 기호 등)가 포함되어 있으므로 데이터 를 문자열로 변환하려면 어떻게해야합니까?


16
UTF-8은 모든 곳에서 UTF-8입니다. UTF-8이되면 플랫폼마다 다른 값이 없습니다. 그것이 요점입니다.
gnasher729

답변:


1155

데이터가 null로 끝나지 않으면 사용해야합니다. -initWithData:encoding:

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

데이터가 null로 끝나는 경우 마지막에 -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로 끝나는 경우 해당 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

2
두 가지 해결책 모두 나를 위해 0을 반환합니다.
Husyn

1
NSData가 null로 종료되는지 여부를 어떻게 알 수 있습니까? Tom Harrington의 답변 : stackoverflow.com/questions/27935054/…을 참조하십시오 . 내 경험상 NSData가 null로 끝나거나 아닌 것으로 가정해서는 안됩니다. 알려진 서버에서도 전송마다 다를 수 있습니다.
엘리스 반 루이

1
@ElisevanLooij 링크 감사합니다. 전송 된 데이터가 무작위로 null로 종료 될 수 있거나 프로토콜이 잘못 정의되어 있다고 주장합니다.
kennytm

28

이 방법을 호출 할 수 있습니다

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

27
데이터가 널 종료 인 경우에만 해당됩니다. 그렇지 않을 수도 있습니다 (실제로는 아닐 수도 있습니다).
Ivan Vučica

나는 왜 지구상에서 이것이 널이 아닌 종료 문자열에서 깨져서 NSData얼마나 많은 바이트를 알고 있는지 알 수 없다 ...
Claudiu

5
@Claudiu, NSData 객체를 전달하지 않고 [data bytes]로 얻은 (const char *)를 전달합니다. 크기는 포인터가 아닙니다. 따라서 가리키는 데이터 블록은 null로 끝나야합니다. 설명서를 확인하십시오.
jbat100

1
@ jbat100 : 물론입니다. 나는 명확하지 않았다. 나는 null이 아닌 종료 NSData에서 NSString(KennyTM의 답변 참조) 로 갈 수 있다는 것을 감안할 때, 그것이 효과 가있는 것이 없다는 것에 놀랐습니다 +(id)stringWithUTF8Data:(NSData *)data.
Claudiu

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를 사용하지 않는 경우 ARC가 필요합니다 autorelease.

이제 끔찍한 장황한 대신 :

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

넌 할 수있어:

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

18

Swift 버전은 String에서 Data로, 다시 String으로 :

Xcode 10.1 • 스위프트 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입니다. 나는 이것이 효과가있는 것으로 나타났습니다 :

목표 -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]; 스위프트 :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];

그리고 매번 예상되는 결과를 얻었습니다.


'쓰레기'결과를 얻은 이유를 <i> 왜 </ i> 이해하는 것이 중요합니다.
Edgar Aroutiounian 5

1

Swift 5에서는 UTF-8을 사용하여 인스턴스를 인스턴스로 변환하기 위해 Stringinit(data:encoding:)이니셜 라이저를 사용할 수 있습니다 . 다음과 같은 선언이 있습니다.DataStringinit(data:encoding:)

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

String주어진 인코딩을 사용하여 주어진 데이터를 유니 코드 문자로 변환 하여 초기화를 반환합니다 .

다음 놀이터 코드는 사용 방법을 보여줍니다.

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.