@autoclosure 和 ?? 讲解

本人摘录来自: 王巍 (onevcat). 《Swifter - 100 个 Swift 必备 Tips (第二版)》

@autoclosure 可以说是Apple 的一个非常神奇的创造,因为这个更多的像是在"hack"这门语言。 简单说, @autoclousure 做的事情就是把一句表达式自动的封装成一个闭包(closure)。 这样有时候在语法上看起来就会非常漂亮

比如我们有一个方法接收一个闭包,当闭包执行的结果为 ture 的时候进行打印:

func loglfTrue(prediccate: () -> Bool){
              if predicate() {
                    print("True")
            }
    }

在调用的时候,我们需要这样写代码
logIfTrue({reture 2 > 1})

当然,在swift中对闭包的用法可以进行一些简化,在着这种情况下我们可以省略吊return,写成:
logIfTrue({2 > 1})

还可以更近一步写成
logIfTrue(2 > 1)

但是不管哪种表达方式,要么时写起来麻烦,要么表达不清晰,看起来都让人不爽。 于是@autoclosure登场了,我们可以改换方法参数,在参数名前面加上 @autoclosure 关键字

    func logIfTure(@autoclosure predicate:() -> Bool){
    if predicate(){
          print("True")
        }
    }

这时候我们可以直接写:
logIfTrue(2 > 1)
来进行调用了, 用swift将会吧 2>1 这个表达式自动转换为()->Bool。这样我们就得到了一个写法简单,表达清楚的式子。

??

在swfit中,有个非常用的操作符,可以用来快速的对nil进行条件判断,那就是??。这个操作符可以判断输入并在当左侧是非nil的Oprational 值时返回其value,当左侧时nil时返回左侧的值,比如:

var lever:Int?
var startLevel = 1
var currentLevel = level ?? startLevel

“在这个例子中我们没有设置过 level,因此最后 startLevel 被赋值给了 currentLevel。如果我们充满好奇心地点进 ?? 的定义,可以看到 ?? 有两种版本:

func ??(optional: T?, @autoclosure defaultValue: () -> T?) -> T?
func ??(optional: T?, @autoclosure defaultValue: () -> T) -> T

在这里我们的输入满足的是后者,虽然表面上看 startLevel 只是一个 Int,但是其实在使用时它被自动封装成了一个 () -> Int,有了这个提示,我们不妨来猜测一下 ?? 的实现吧:”

“func ??(optional: T?, @autoclosure defaultValue: () -> T) -> T {
    switch optional {
        case .Some(let value):
            return value
        case .None:
            return defaultValue()
        }
}”

“可能你会有疑问,为什么这里要使用 autoclosure,直接接受 T 作为参数并返回不行么?这正是 autoclosure的一个最值得称赞的地方。如果我们直接使用 T,那么就意味着在 ?? 操作符真正取值之前,我们就必须准备好一个默认值,这个默认值的准备和计算是会消耗性能的。但是其实要是 optional 不是 nil 的话,我们是完全不需要这个默认值,而会直接返回 optional解包后的值。这样一来,默认值就白白准备了,这样的开销是完全可以避免的,方法就是将默认值的计算推迟到 optional 判定为 nil之后。
就这样,我们可以巧妙地绕过条件判断和强制转换,以很优雅的写法处理对 Optional 及默认值的取值了。最后要提一句的是,@autoclosure 并不支持带有输入参数的写法,也就是说只有形如 () -> T 的参数才能使用这个特性进行简化。另外因为调用者往往很容易忽视 @autoclosure 这个特性,所以在写接受 @autoclosure 的方法时还请特别小心,如果在容易产生歧义或者误解的时候,还是使用完整的闭包写法会比较好。”

你可能感兴趣的:(@autoclosure 和 ?? 讲解)