Swift 基于 willSet & didSet 的订阅block(Observable)

本文原创 作者:丛瑜帅 如需转载请注明 (2017-06-06)

起因:

MVC一直以来是代码组织架构中苹果公司所推崇的开发模式,但由于工程类文件的日益增多,MVC中C层(Controller层)的日益臃肿,产品需求复杂化,迭代速度越来越快,老架构设计已经渐渐跟不上高强度的组件化团队化开发了。最近一直在寻求一种开发模式,能让多个团队成员可以同时开发且逻辑清晰。期间阅读了很多文章,比如VIPER架构、UBer公司未开源的Riblets架构、MVVM架构等,最终决定自己针对MVVM进行一次架构改造,并加入VIPER的特点。其中MVVM的ViewModel的轻实现,当下被列为攻坚环节。

MVVM的ViewModel中采用KVO的观察者模式监听,调用ViewController来进行整个架构的解耦设计。在Objective-C当中得益于强大的Runtime机制可以实现对任意类型的观察者监听。虽然Objective-C中可以任意定义KVO,但是经历过大项目的朋友一定首先会想到Objective-C中的KVO在使用的轻便型上差强人意,需要addObserver和removeObserver,且如果Context上下文弄错了,会有一定的崩溃风险,这是需要深刻了解Objective-C的释放避免指针的循环引用等。

Swift作为一个静态编译型语言,它摒弃了Objective-C中的Runtime机制。想要开启动态Property需要再Swift的Property前面增加声明:dynamic,且使用dynamic必须是基于NSObject基类所构造的类型,这样做必然会丧失对Swift原始数据类型的支持,可见其是不好的。而且预计没有多少朋友记得给变量打上dynamic的标记吧,起码我不会

很庆幸的是Swift语言在自己的Property中增加了getter/ setter的属性观察器,并对setter的属性观察器提供了willSet / didSet的两个观察器来详细监听值的变化。这让我们看到Swift本身是汲取了Objective-C在Runtime中创造的经验和灵感,并将观察者模式轻量化,以相当优雅的方式去表示一个值的变化过程。

class valueDemo {
    var value:String = "" {
        willSet {
            print("newValue:", newValue)
        }
        didSet {
            print("done:", value)
        }
    }
}

可是我们在开发中不仅仅是这样的简单环境,我们需要针对MVVM中ViewModel开放一个被观察者连接给ViewController,两者产生联动。此时有人想到:"我提供一个闭包(block)设置给didSet就好了呀"。确实你可以这样做。为每一个Property提供一个block虽然可行,但没有重用好这一机制是则会让代码变得重复。那我们就要寻找一个好一点的方法来能让Property变成一个被观察者,当它发生变更的时候,触发一批block回调。

分析第三方:

ReactiveCocoa和RxSwift的第三方库来实现是可以很好地实现观察者模式(笔者更喜欢后者RxSwift的书写风格)。确实,现在MVVM中采用RxSwift解耦作为中间件确实是产品开发潮流,这就像某种服装搭配趋势一样的流行。那问题随之而来,采用ReactiveCocoa和RxSwift都哪些共同缺点呢?我们开发实战的时候肯定会遇到下面的问题:

  1. 订阅和分发导致它本身的执行效率低,会有大量的触发栈和循环去进行订阅消息的分发,遍历逐个投递。
  2. Swift本身的语法导致从Swift v.2 -> v.3 -> v.4的语法升级受制于苹果的语言规则。Swift语言开发者的开发理念是快速激进式的开发(我给它定名为:语法摧毁),虽然xcode提供了自动化转换语法功能,但难免会有转换错误和手动修改的情况。这样对于我们程序本身是非常不稳定的变化,导致我们出现重写程序组件的问题,甚至摧毁式的无法编译
  3. ReactiveCocoa和RxSwift的开发成本比较高,语法体系“奇特”(碎片化的代码,打散业务逻辑,由第三方库限定语法编写方式),导致团队间在合作时逻辑代码理解难度加大。团队成员间的代码沟通变慢。如果团队加入新人,学习成本则会提高。
  4. 库文件升级缓慢,受制于他人,如果停止更新,可能你的产品就要赶紧寻找其他第三方库来进行重构。

基于以上几点缺点,我在这里不赞同采用这样的第三方组件的开发方式开发,虽然它们很酷炫、显得高大上!

全新创建:

那难道没有一个又轻又容易维护的观察者模式吗?答案是有的!
那我们就从零开始一步步实现一个基于Swift 3~4的低调奢华有内涵的观察者模式(题外话由于我所书写的日期是2017-6-6,正好是Swift 4发布当日,我的工程文件又一次被Swift4的升级所摧毁,被摧毁的是第三方库,那我还是自己造一个轮子吧!)

先来描述一下基本原理:

  1. 实现一个用于产生被观察者的自定义泛型类:Observable<T>
  2. Observable自身提供blocks的闭包数组存放订阅者的闭包
  3. 基于Observable中的value的setter方法,手动调用每个闭包

先来看一下基础代码:

// 需要持有一批blocks,则必须创建一个类作为空间
class Observable<T> {
    typealias ObservableBlock = (T) -> ()
    private var blocks: [ObserverBlock] = []  // 持有blocks
    
    init(_ t:T) { self.value = t }  // 初始化value
    var value:T {
        didSet {
            // 实现didSet来遍历block,触发回调
            for block in blocks {
                block(self.value)
            }
        }
    }

    // 订阅
    func subscribe(block:@escaping ObserverBlock) {
        blocks.append(block)
    }
}

run exmple:

let example = Observable<String>("")
example.subscribe { (newValue:String) in
    print("newValue:", newValue)
}
example.value = "a"
example.value = "b"

代码的运行结果:

newValue: a
newValue: b

看到运行结果,很不错!基于简单blocks持有,基于didSet就可以完成对于一个变量设置的变更监听。

继续完善

仔细打量了代码,中间缺少几个能力:

  1. 如何将example.value = "a"的写法,将开发者的敲击键盘所消耗的卡路里降到最低呢。赋值形式换为:example <- "a"
    这里想到了Swift的《高级运算符重载》:【https://www.cnswift.org/advanced-operators#spl-17
  2. 缺少删除订阅者block能力。这个能力需要在订阅时将订阅者传递给Observable加以持有,并提供unSubscribe方法

第一步我们先来加入高级运算符重载,片段代码:

infix operator <-: ObservableChange 
precedencegroup ObservableChange {
    associativity: left                 // 表示左结合
}
public func <- <T> (left: Observable<T>, right: T) {
    left.value = right
}

完整代码:<a name="block_observable">[纯block,可自动释放内存]</a>

// 高级运算符重载必须声明在final顶级访问级别的类中
public final class Observable<T> {
    typealias ObserverBlock = (T) -> ()
    private var blocks: Array<ObserverBlock> = Array()
    
    init(_ t:T) { self.value = t }
    var value:T {
        didSet {
            for block in blocks {
                block(self.value)
            }
        }
    }
    func subscribe(block:@escaping ObserverBlock) {
        blocks.append(block)
    }
    deinit {
       print("Observable", #function)
    }
}

/* 
定义 <- 运算符 
运算符定义必须放在文件级别当中
*/
infix operator <-: ObservableChange
precedencegroup ObservableChange {
    associativity: left                 // 表示左结合
}

public func <- <T> (left: Observable<T>, right: T) {
    left.value = right
}

run exmple :

let example = Observable<String>("")
example.subscribe { (newValue:String) in
     print("newValue:", newValue)
}
example.value = "a"
example.value = "b"
example <- "a"

代码的运行结果:

newValue: a
newValue: b
newValue: a

重载看上去还不错,很精简!那继续完善,填补后续的功能

第二步添加unSubscribe方法

起初我想直接通过block闭包的相等性检查,通过block闭包相等,来移除blocks中的指定闭包,但是失败了。比如代码:

public final class Observable<T> {
    typealias ObserverBlock = (T) -> ()
    private var blocks: Array<ObserverBlock> = Array()
    
    init(_ t:T) { self.value = t }
    var value:T {
        didSet {
            for block in blocks {
                block(self.value)
            }
        }
    }
    func subscribe(block:@escaping ObserverBlock) {
        blocks.append(block)
    }
    
    // 移除订阅
    func unSubscript(block:@escaping ObservableBlock) {
        var blocksFiltered = blocks.filter { (blockInArray:ObservableBlock) -> Bool in
            return blockInArray !== block  // !!!!!!!无法编译,编译报错!!!!!!!
            //报错信息:  Cannot check reference equality of functions;operands here have type '(T)->()' and '(T)->()'
        }

        self.blocks = blocksFiltered
    }
}

看到//报错信息: Cannot check reference equality of functions;operands here have type '(T)->()' and '(T)->()'
发现Swift中是不允许将两个闭包进行的比较的。虽然遗留的C API中是有unsafeBitCast可以对两个闭包进行比较,但我还是放弃这样的写法。

unsafeBitCast 相关使用:https://stackoverflow.com/questions/24111984/how-do-you-test-functions-and-closures-for-equality

那既然block无法比较相等,就只能讲上下文与blocks进行绑定关系,来实现订阅和删除订阅。

// 定义高级运算符重载,必须为final访问权限的声明
public final class Observable<T> {
    typealias ObserverBlock = (_ oldValue:T, _ newValue:T) -> ()    // 订阅block,增加old和new的传值
    typealias ObserverEntry = (observer: AnyObject, block: ObserverBlock)   // 观察者元组
    private var observers: [ObserverEntry]  // 观察者Array
    
    init(_ value:T) {
        self.value = value
        observers = []
    }
    var value:T {
        didSet {
            observers.forEach { (entry: ObserverEntry) in
                let (_, block) = entry
                block(oldValue, value)
            }
        }
    }
    
    // 订阅,创建观察者元组
    func subscribe(observer:AnyObject, block:@escaping ObserverBlock) {
        observers.append(ObserverEntry(observer:observer, block:block))
    }
    
    // 解除订阅,根据元组中的观察者移除
    func unSubscribe(observer:AnyObject) {
        let filtered = observers.filter { (entry: ObserverEntry) in
            let (owner, _) = entry
            return owner !== observer
        }

        observers = filtered
    }
}

infix operator <-: ObservableChange
precedencegroup ObservableChange {
    associativity: left                 // 表示左结合
}

// 运算符重载
public func <- <T> (left: Observable<T>, right: T) {
    left.value = right
}

run example:

let example = Observable<String>("")
example.subscribe(observer: self) { (oldValue:String, newValue:String) in
    print("oldValue:", oldValue, "newValue:", newValue)
}
example.value = "a"
example.value = "b"
example <- "a"
example.unSubscribe(observer: self)
example <- "c"  // 取消订阅,则不会看到"c"的打印

代码的运行结果:

oldValue:  newValue: a
oldValue: a newValue: b
oldValue: b newValue: a
// 这里没有看到“c”

已知弊端:

不过本观察者订阅模式和其他的第三方组件其实都有弊端:

  1. 就是插入式编程,
  2. 内存循环应用
    插入式编程就是会将原有的代码的变量类型破坏,从而让类型都趋向于Observable<T>数据类型,这样喜欢纯正变量监听的话,当下除了willSet和didSet,尚未发现其他更优雅的方法!

*** 而内存循环应用,需要将被保存在entry当中的Observer在必要的时候unSubscribe掉才可以解决循环引用的问题。

2017-06-12后续

经过测试我采用了自动释放和手动释放两个方式编写Observable源码。而上面的代码中,我将subscribe:Observer修改为block与ObserverName绑定的形式,来解决内存循环引用的问题。
进一步修改代码我们来看一下:

// final class for operator <-
// 高级运算符重载必须声明在final顶级访问级别的类中

public final class Observable<T> {
    typealias ObserverBlock = (_ oldValue: T, _ newValue: T) -> ()
    typealias ObserversEntry = (block: ObserverBlock, observerName:String?)
    private var observers: Array<ObserversEntry>

    init(_ value: T) {
        self.value = value
        observers = []
    }
    
    var value: T {
        didSet {
            observers.forEach { (entry: ObserversEntry) in
                let (block, _) = entry
                block(oldValue, value)
            }
        }
    }

    func subscribe(block: @escaping ObserverBlock) -> Self {
        let entry: ObserversEntry = (block: block, nil)
        observers.append(entry)
        return self
    }
    
    // set ObserverName for unsubscribe
    func addObserverName(_ observerName: String) {
        if observers.count > 0 {
            observers[observers.count-1].observerName = observerName
        }
    }
    
    // remove subscribe with ObserverName
    func unSubscribe(_ observerName: String) {
        let filtered = observers.filter { entry in
            let (_, observerNameSaved) = entry
            if (observerNameSaved != nil) {
                return observerNameSaved != observerName
            } else {
                return true
            }
        }
        
        observers = filtered
    }
}
/*
 定义 <- 运算符
 运算符定义必须放在文件级别当中
 */
infix operator <-: ObservableChange
precedencegroup ObservableChange {
    associativity: left                 // 表示左结合
}

public func <- <T> (left: Observable<T>, right: T) {
    left.value = right
}

看到上方最新的代码,我们可以观察到添加了一个addObserverName(_ observerName:)用于给订阅block注册关键字,这样就可以将一批订阅者注册并取消订阅。且没有内存引用问题。思路我借鉴了RXSwift的调用时机:addDisposeBag(disposebag:)
那么释放的使用方法如下:

let example = Observable<String>("")
example.subscribe { (old:String, new:String) in
    print("oldValue:", old, "newValue:", new)
}.addObserverName("TheExampleName")
example <- "a"
example.unSubscribe("TheExampleName")

好了,经过细细打磨的Observable已经初步具备了观察者能力了,并且可以轻巧的应用于变量的观察
全部代码:
https://github.com/maxcong/Observable-Block-Swift

我在编写期间试用了google的一个开发者开发的Observable-Swift的,
但这个只针对于Swift 3,功能略复杂了,最后放弃

此文抛砖引玉,希望看到的开发者如果有优雅的方法可以在文章后面留言。深表感谢!

本文原创 作者:丛瑜帅 如需转载请注明

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

推荐阅读更多精彩内容

  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 12,071评论 4 62
  • 记 往昔 旧故里 充满回忆 少年游经历 至今留恋不已 天真岁月多甜蜜 追蝶逐梦童言无忌 然今夕何夕流年离弃 纵与世...
    一袍风阅读 304评论 4 1
  • 某一天的某个瞬间对日历突然有了好感,似乎当时的自己比现在忙碌一些,觉得写下的事情就是提上了日程。一度徜徉,有时候也...
    一座云阅读 247评论 0 0
  • 1. “寻找罗姓先生,黄色人种,烟灰毛衫,发茂盛,体...
    梦呓岛的鱼小姐阅读 382评论 1 0