Swift 为什么String.addingPercentEncoding()的返回值是可选的?

Swift 为什么String.addingPercentEncoding()的返回值是可选的?,swift,cocoa,unicode,utf-8,url-encoding,Swift,Cocoa,Unicode,Utf 8,Url Encoding,百分比转义的字符串方法的签名为: func addingPercentEncoding(withAllowedCharacters: CharacterSet) -> String? (这是通过在Swift 2中添加带有允许字符的百分比编码来实现的string。) 为什么这个方法返回一个可选的 文档中说,“如果转换不可能”,该方法返回nil,但不清楚在什么情况下转义转换可能失败: 使用UTF-8对字符进行转义,UTF-8是一种完整的Unicode编码。任何有效的Unicode字

百分比转义的
字符串
方法的签名为:

func addingPercentEncoding(withAllowedCharacters: CharacterSet)
    -> String?
(这是通过在Swift 2中添加带有允许字符的百分比编码来实现的
string。)

为什么这个方法返回一个可选的

文档中说,“如果转换不可能”,该方法返回nil,但不清楚在什么情况下转义转换可能失败:

  • 使用UTF-8对字符进行转义,UTF-8是一种完整的Unicode编码。任何有效的Unicode字符都可以使用UTF-8编码,因此可以转义

  • 我认为该方法可能对允许的字符集和用于转义的字符之间的不良交互应用了某种健全性检查,但事实并非如此:无论允许的字符集是否包含“%”,该方法都会成功,如果允许的字符集为空,该方法也会成功


目前看来,非可选返回值似乎在强制进行无意义的错误检查。

我向苹果公司提交了一份关于这一问题的错误报告,并得到了回复——回复非常有帮助,至少

事实证明(让我大吃一惊)可以成功地创建包含无效Unicode的未配对UTF-16代理字符形式的Swift字符串。这样的字符串可能导致UTF-8编码失败。下面是一些说明此行为的代码:

// Succeeds (wat?!):
let str = String(
    bytes: [0xD8, 0x00] as [UInt8],
    encoding: String.Encoding.utf16BigEndian)!

// Returns nil:
str.addingPercentEncoding(withAllowedCharacters:
    CharacterSet.alphanumerics)

基于Paul Cantrell的回答,这是一个小演示,表明同样的方法也可以在Objective-C中返回null,尽管String和NSString在编码方面是不同的:

uint8_t字节[2]={0xD8,0x00};
NSString*string=[[NSString alloc]initWithBytes:字节长度:2编码:NSUTF16BigEndianStringEncoding];
//\ud800
NSLog(@“%@”,字符串);
NSString*escapedString=[string stringByAddingPercentEncodingWithAllowedCharacters:NSCharacterSet.URLHostAllowedCharacterSet];
//(空)
NSLog(@“%@”,转义字符串);

为了好玩,将以相同的方式:

%20convertUTF162Char%3A%20low%20urrogate%20expected%2C%20b%3D0%21%00中出现错误%20