Optional chaining的秘密(一)

昨天看了一位简书朋友写的内容<<Swift超基础语法(可选链篇)>>,我产生了一个小的疑问,为什么可选链中最后一个访问的属性name并不是Optional的,但在可选连中会返回一个Optional的值,这个封箱操作是在什么时候完成的呢?所以我在谷歌上翻箱倒柜的找到了这个答案。

定义一个可以实现Optional chaining的操作符

我们可以把 ?. 作为Optional chaining的操作符来思考,?.的属性包括:

  • 接收两个参数,一个Optional值,和一个可以接收unwrapped value作为参数的function
  • 返回一个Optional值

我们来定义一个操作符,让它来可以替代?.完成Optional chaining操作,通过这个操作符的定义我们来探索一下Optional chaining的秘密。根据 ?. 的属性,我们确定我们的新操作符应该定义为:

operator infix |- { associativity left }

@infix |-<T,U> (T?, f: T -> U?) -> U?

|- 接收两个参数,第一个是一个可选类型,第二个是一个函数。

@infix |-<T,U> (opt:T?, f: T -> U?) -> U? {
    switch opt {
    case .Some(let x):
        return f(x)
    case .None:
        return .None
    }
}

验证操作符

现在如果我们打开playgroud,把下面的代码片段copy进去,那么它可以完美的工作了。

public class Demo {
    public let subDemo:SubDemo?
    init(subDemo sDemo:SubDemo? = nil) {
        self.subDemo = sDemo
    }
}

public class SubDemo {
    public let count:Int = 1
}

let aDemo:Demo? = nil
let bDemo:Demo? = Demo()
let cDemo:Demo? = Demo(subDemo: SubDemo())

let aCount = aDemo |- { $0.subDemo } |- { $0.count } // {None}
let bCount = bDemo |- { $0.subDemo } |- { $0.count } // {None}
let cCount = cDemo |- { $0.subDemo } |- { $0.count } // {Some 1}

问题的原因

在这个代码片段中,我们可以看到 |- 可以像 ?. 那样实现Optional chaining。这里就到了我们最开始的问题:为什么链条上的最后一个属性count会被返回一个Optional类型,这种操作其实并不需要我们手动进行处理,原因就是 |- 操作符的闭包参数 f: T -> U? 返回的是一个可选类型。swift会将返回值自动打包。所以我们一定会得到一个可选值。

再进一步

我们知道map函数可以对Optional类型值进行安全访问,<<optional的九种拆包方式,你知道几种?>>

let optional: Int? = 4
optional.map{ print($0) }

怎么解释这个map解包操作呢?原因是Optional中实现了一个map方法,我们可以在Optional中实现一个flatMap方法,来实现同样的能力。

extension Optional {
    func flatMap<Z>(f:T->Z?) -> Z? {
        switch self {
        case .Some(let a):
            return f(a)
        case .None: 
            return .None
        }
    }
}

我们看到 flatMap 可以接收一个闭包,它首先对Optional进行拆包工作,然后将获得的值传递给闭包进行处理。于是我们之前的代码可以改为通过flatMap实现。

let aCount = aDemo.flatMap { $0.subDemo }.flatMap { $0.count } // {None}
let bCount = bDemo.flatMap { $0.subDemo }.flatMap { $0.count } // {None}
let cCount = cDemo.flatMap { $0.subDemo }.flatMap { $0.count } // {Some 1}

总结

话题未完,待续。

参考内容:

Understanding Optional Chaining

Meet Optional

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 对各种值为"空"的情况处理不当,几乎是所有Bug的来源。 在我们的例子里,尽管tmp的值是nil,但调用tmp的r...
    AKyS佐毅阅读 10,594评论 1 13
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 135,168评论 19 139
  • 国家电网公司企业标准(Q/GDW)- 面向对象的用电信息数据交换协议 - 报批稿:20170802 前言: 排版 ...
    庭说阅读 11,262评论 6 13
  • 01 80后中年人 前几天,人民日报官方微博发出一条消息,标题为80后们:“中年人”这个称谓,你们真的接受吗? 正...
    Ragingfire烈火阅读 2,407评论 2 1
  • 手机震动出你的声音 我听到了夏天 早上七点 七月比我醒得更早 能闻到一场雨的靠近 我等待着 就像等一通很短的电话 ...
    郭小台阅读 379评论 0 1