将UTF-8编码的NSData转换为NSString

Ash*_*kar 562 nsstring nsdata ios

NSData从Windows服务器编码UTF-8 ,我想将其转换NSString为iPhone.由于数据包含在两个平台上具有不同值的字符(如度数符号),如何将数据转换为字符串?

ken*_*ytm 1148

如果数据不是以null结尾,则应使用 -initWithData:encoding:

NSString* newStr = [[NSString alloc] initWithData:theData encoding:NSUTF8StringEncoding];
Run Code Online (Sandbox Code Playgroud)

如果数据以空值终止,则应该使用-stringWithUTF8String:以避免最后的额外数据\0.

NSString* newStr = [NSString stringWithUTF8String:[theData bytes]];
Run Code Online (Sandbox Code Playgroud)

(请注意,如果输入的UTF-8编码不正确,您将获得nil.)


Swift变种:

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

如果数据以空值终止,您可以通过删除该空字符的安全方式或类似于上述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:))
Run Code Online (Sandbox Code Playgroud)

  • 明白这一点:当对非空终止的字符串使用"stringWithUTF8String:"时,结果是不可预测的! (31认同)
  • 小心!!如果使用stringWithUTF8String,不要传递NULL参数,否则会抛出异常 (5认同)
  • 两种解决方案都归零给我. (2认同)

Gou*_*dsc 28

你可以调用这个方法

+(id)stringWithUTF8String:(const char *)bytes.
Run Code Online (Sandbox Code Playgroud)

  • 仅当数据以空值终止时.它可能不是(事实上,可能不是). (27认同)
  • @Claudiu,你没有传入一个NSData对象,你传递的是一个用[data bytes]获得的(const char*),它只是一个指针,没有大小信息.因此,它指向的数据块必须为空终止.查看文档,它明确说明. (4认同)

Cla*_*diu 19

我谦卑地提交一个类别,以减少烦恼:

@interface NSData (EasyUTF8)

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

@end
Run Code Online (Sandbox Code Playgroud)

@implementation NSData (EasyUTF8)

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

@end
Run Code Online (Sandbox Code Playgroud)

(请注意,如果您不使用ARC,则需要使用ARC autorelease.)

而不是令人震惊的冗长:

NSData *data = ...
[[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
Run Code Online (Sandbox Code Playgroud)

你可以做:

NSData *data = ...
[data asUTF8String];
Run Code Online (Sandbox Code Playgroud)


Leo*_*bus 18

Swift版本从String到Data再返回String:

Xcode 10.1•Swift 4.2.1

extension Data {
    var string: String? {
        return String(data: self, encoding: .utf8)
    }
}
Run Code Online (Sandbox Code Playgroud)
extension StringProtocol {
    var data: Data {
        return Data(utf8)
    }
}
Run Code Online (Sandbox Code Playgroud)
extension String {
    var base64Decoded: Data? {
        return Data(base64Encoded: self)
    }
}
Run Code Online (Sandbox Code Playgroud)

操场

let string = "Hello World"                                  // "Hello World"
let stringData = string.data                                // 11 bytes
let base64EncodedString = stringData.base64EncodedString()  // "SGVsbG8gV29ybGQ="
let stringFromData = stringData.string                      // "Hello World"
Run Code Online (Sandbox Code Playgroud)
let base64String = "SGVsbG8gV29ybGQ="
if let data = base64String.base64Decoded {
    print(data)                                    //  11 bytes
    print(data.base64EncodedString())              // "SGVsbG8gV29ybGQ="
    print(data.string ?? "nil")                    // "Hello World"
}
Run Code Online (Sandbox Code Playgroud)
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"
Run Code Online (Sandbox Code Playgroud)


mik*_*eho 16

有时,其他答案中的方法不起作用.在我的情况下,我使用我的RSA私钥生成签名,结果是NSData.我发现这似乎有效:

Objective-C的

NSData *signature;
NSString *signatureString = [signature base64EncodedStringWithOptions:0];
Run Code Online (Sandbox Code Playgroud)

迅速

let signatureString = signature.base64EncodedStringWithOptions(nil)
Run Code Online (Sandbox Code Playgroud)