我的类有一个NSURL
类型的属性,它是从字符串初始化的。该字符串在编译时已知。
要使类正常运行,必须在初始化(不是更晚)时将其设置为其预期值,因此将其定义为可选(隐式展开或其他方式)没有意义:
class TestClass: NSObject {
private let myURL:NSURL
...
假设NSURL(string:)
(返回NSURL?
)如果传递了编译时已知的有效URL字符串将永远不会失败,我可以这样做:
override init() {
myURL = NSURL(string: "http://www.google.com")!
super.init()
}
但是,我在某种程度上对强制解包感到不舒服,并希望以某种方式保护URL初始化。如果我试试这个:
guard myURL = NSURL(string: "http://www.google.com") else {
fatalError()
}
可选类型'NSURL?'的值没有打开;你的意思是用'!'要么 '?'?
(注意:没有办法在上面的代码中的任何地方添加!
或?
来修复错误。有条件的解包只发生在guard let...
guard var...
,并且已经定义了myURL
)
我理解为什么会失败:即使成功调用NSURL(string:)
也会返回包含在可选NSURL
中的(有效)NSURL?
,所以在分配给myURL
之前我仍然需要以某种方式解开它(这是非可选的,因此不能兼容-is)。
我可以通过使用中间变量解决这个问题:
guard let theURL = NSURL(string: "http://www.google.com") else {
fatalError()
}
myURL = theURL
......但这显然不是优雅的。
我该怎么办?
更新另一种不使用guard
的方法是使用switch
,作为选项映射到Optional
枚举:
init?() {
switch URL(string: "http://www.google.com") {
case .none:
myURL = NSURL()
return nil
case let .some(url):
myURL = url
}
}
虽然你仍然得到一个url
局部变量。
原始答案
您可以将初始化程序声明为可用的初始化程序,并在url字符串解析失败时返回nil
,而不是抛出致命错误。这将使客户端更清楚您的类初始化程序可能在某些时候失败。不过,你仍然不会摆脱guard
。
init?() {
guard let url = URL(string: "http:www.google.com") else {
// need to set a dummy value due to a limitation of the Swift compiler
myURL = URL()
return nil
}
myURL = url
}
这在调用者方面增加了一点复杂性,因为它需要检查对象创建是否成功,但是如果对象初始化器可能无法构造对象,则它是推荐的模式。您还需要放弃NSObject
继承,因为您无法使用可用版本(init
)覆盖init?
。
您可以在Swift blog,Apple's documentation或SO question上找到有关可用初始化器的更多详细信息。