Swift标准库源码之旅 - LazySequence

背景

Lazy sequences can be used to avoid needless storage allocation and computation, because they use an underlying sequence for storage and compute their elements on demand.

在使用数组或某一序列的map/filter等方法的时候,容器会立刻进行遍历对所有元素执行transformisIncluded闭包,时间复杂度是O(n),并且返回的新数组也会占用对应内存空间。如果transform里的操作比较耗时,容器元素比较多,那么此种方式的时间消耗就不能忽略了。

举个极端的例子:

       let result1 = Array(repeating: 3, count: 99).map { (num) -> String in
            (0 ... 9999).reduce("0") { (current, num) in
                return current + num.description
            }
        }

上述代码的执行时间需要5.8s. 即使我们还根本没有用到其中的元素.

Lazy的方式就是将transform等需要遍历元素的操作进行延时,在需要元素的时候再进行transform.

思路

transform闭包作为属性保存下来,重写迭代方法,在迭代之前元素的基础上进行·transform·返回。

实现 LazyMapSequence

所有的Sequence皆可进行lazy. 为Sequence类型添加扩展方法

extension Sequence {
 var lazy: LazySequence<Self> {
    return LazySequence(_base: self)
  }
}

LazySequence是一个中间类型,把原始Sequence作为属性保存起来备用.后面的map/filter都在这个类型基础上扩展

struct LazySequence<Base: Sequence>: Sequence {
    typealias Element = Base.Element
    
    var _base: Base
    init(_ base: Base) {
        self._base = base
    }
    
    func makeIterator() -> Base.Iterator {
        return _base.makeIterator()
    }
}

LazySequence提供map方法,返回值是一个重写了迭代方法的新的Sequence

extension LazySequence {
    func map<ResultElement>(_ transform: @escaping (Element) -> ResultElement) -> LazyMapSequence<Base, ResultElement> {
        LazyMapSequence(self._base, transform: transform)
    }
}

他要把transform和原始序列保存下来.

struct LazyMapSequence<Base: Sequence, Element>: Sequence {
    
    typealias Element = Element
        
    var _base: Base
    var transform: (Base.Element) -> Element
    
    init(_ base: Base, transform: @escaping (Base.Element) -> Element) {
        self._base = base
        self.transform = transform
    }
    
    func makeIterator() -> Iterator {
        Iterator(_base.makeIterator(), transform: transform)
    }
}

他的迭代方法如下,取base的元素然后进行transform返回

extension LazyMapSequence {
    struct Iterator: MyIteratorProtocol {
        var _base: Base.Iterator
        var transform: (Base.Element) -> Element
        
        init(_ base: Base.Iterator, transform: @escaping (Base.Element) -> Element) {
            self._base = base
            self.transform = transform
        }
        
        mutating func next() -> Element? {
            return _base.next().map(transform)
        }
    }
}

以上仅解决了Sequence进行map延迟遍历的问题.
更常用的情况是一个数组map后使用下标取值.也需要支持.

也就是当LazyMapSequenceBase是个Collection的时候

public typealias LazyMapCollection<T: Collection,U> = LazyMapSequence<T,U>

然后就可以给LazyMapCollection类型扩展方法了

extension LazyMapCollection: Collection {
  ...
  ...
  subscript(position: Base.Index) -> Element {
    return _transform(_base[position])
  }
}

这样,数组类型的lazy后的map方法返回的LazyMapSequence就支持了下标访问.transform只在取某一值的时候调用.

上面只完成了map方法的lazy. 对于Filter方法想要实现lazy同样需要已基本相同的方式写一个LazyFilterSequence写其对应的迭代方法.

## 实现 LazyFilterSequence

extension LazySequence {
    func filter(_ isIncluded: @escaping (Element) -> Bool) -> LazyFilterSequence<Base> {
        LazyFilterSequence(_base, isIncluded: isIncluded)
    }
}

struct LazyFilterSequence<Base: MySequence>: MySequence {
    
    typealias Element = Base.Element
        
    var _base: Base
    var _predicate: (Base.Element) -> Bool
    
    init(_ base: Base, isIncluded: @escaping (Base.Element) -> Bool) {
        self._base = base
        self._predicate = isIncluded
    }
    
    func makeIterator() -> Iterator {
        Iterator(_base.makeIterator(), isIncluded: _predicate)
    }
}

extension LazyFilterSequence {
    struct Iterator: MyIteratorProtocol {
        var _base: Base.Iterator
        var _predicate: (Base.Element) -> Bool
        
        init(_ base: Base.Iterator, isIncluded: @escaping (Base.Element) -> Bool) {
            self._base = base
            self._predicate = isIncluded
        }
        
        mutating func next() -> Element? {
            while let value = _base.next() {
                if _predicate(value) {
                    return value
                }
            }
            return nil
        }
    }
}

LazyMapSequence实现相似,不在赘述。

但是Filter是筛选元素得到一个新结果,数组的下标访问如何做到lazy呢? 不执行完Filter是不能拿到最后数组个数的吧。带着这个猜想发现了有意思的情况.

        let result = Array(repeating: 100, count: 99999).lazy.filter { $0 > 100 }
        print(result.count) // 0
        print(result[3]) // 100

这个筛选结果必然是空的.count为0是没问题的. 但是下标取值[3]确是返回了100

去看一下源码发现下标取值的方法是取的原数组的值, 并没有经过筛选

subscript(position: Index) -> Element {
    return _base[position]
}

所以下面这种操作方式也会得到你意想不到的结果.

        var array: [String?] = ["a", nil, nil, "c"]
        let result = array.lazy.filter { $0 != nil }
        for i in 0 ..< result.count {
            print(result[i])   // a, nil
        }

result本来是筛选过的不为nil的结果,期望值应该是"a"和"c". 结果却是"a"和"nil"

有一些方法没有新生成一个LazyxxSequence而是建立在上面两个Sequence之上的.

Lazy的CompactMap

func compactMap<ElementOfResult>(
    _ transform: @escaping (Elements.Element) -> ElementOfResult?
  ) -> LazyMapSequence<
    LazyFilterSequence<
      LazyMapSequence<Elements, ElementOfResult?>>,
    ElementOfResult
  > {
    return self.map(transform).filter { $0 != nil }.map { $0! }
  }

它是由LazyFilterSequenceLazyMapSequence组合实现的方法.因为用到了Filter. 下标取值仍然是取得原Collection的值.
但是自己实现时候发现LazyMapSequence类型其实并没有Filter方法, LazyFilterSequence方法也并没有map方法. 手动给具体类型加上吗? 但后面还有LazyDropWhileSequence类型,想要自由组合的话这种方式太硬了一点..
让他们共同遵循一个协议,给此协议添加扩展方法就可以让所有类型都具有map/filter

protocol LazySequenceProtocol: MySequence {
    
}
extension LazySequenceProtocol {
    func filter(_ isIncluded: @escaping (Element) -> Bool) -> LazyFilterSequence<Self> {
        LazyFilterSequence(self, isIncluded: isIncluded)
    }
}
extension LazySequenceProtocol {
    func map<U>(_ transform: @escaping (Element) -> U) -> LazyMapSequence<Self, U> {
        .init(self, transform: transform)
    }
}

接下来就有了上面CompactMap返回的多个尖括号的泛型嵌套类型。好处是可以自由组合。

为什么LazySequenceProtocol里没有定义方法,其实Swift源代码里是有的,但是笔者目前没有弄明白里面方法的意义。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,772评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,458评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,610评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,640评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,657评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,590评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,962评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,631评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,870评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,611评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,704评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,386评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,969评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,944评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,179评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,742评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,440评论 2 342