RxSwift-中介者模式

proxy.png

中介者模式,顾名思义,通过中介来连接买家和供应商,减少买家和供应商的联系成本。在RxSwift中存在很多中介者来帮我们处理很多事情,如map来帮我们处理数据并转化为新的序列;filter来帮我们筛选数据并产生新序列;zip来帮助我们将多个序列合成为一个序列。这些内部复杂的实现不可能每次在用到时重新实现一边,通过中介者达到一个很好的复用及管理。

map

Observable<Int>.of(1,2,3,4,5,6)
    .map{
        $0+10
    }
    .subscribe(onNext: { (val) in
        print(val)
    }).disposed(by: disposeBag)
  • 输出结果在原有序列元素的基础上+10
  • 输出结果为:11 12 13 14 15 16

filter

Observable<Int>.of(1,2,3,4,5)
    .filter {$0>4}
    .subscribe(onNext: { (val) in
        print(val)
    }).disposed(by: disposeBag)
  • filter中介者筛选条件,筛选数据
  • 输出结果为:5

以上RxSwift操作符均是我们的中介者,下面我们来看一下定时器中介者的演化。

普通实现

实现一个定时器并打印:

self.timer = Timer.init(timeInterval: 1, target: self, selector: #selector(timerFire), userInfo: nil, repeats: true)
RunLoop.current.add(self.timer!, forMode: .common)

@objc func timerFire() {
    print("timer")
}

deinit {
    print("\(self.classForCoder) 销毁")
}

运行打印,能够跑起来,到这很多人觉得,这不就行了吗,有定时器直接用就好。我们来检查一下,界面是否能够正常释放。这里说明当前页面是push进来的页面,点击返回即可。

返回后发现我们的deinit并没有消失,那就说明当前页面出现了循环引用,那么此处循环引用肯定是在timerself之间的。

  • self持有timertimer持有当前self构成循环引用

当然有人会想到,在deinit中给timer置空不就打破了吗,可以吗?当然不行,self不能释放就不会执行deinit。那如果在页面快消失的时候使定时器失效,并置空呢?如下:

override func viewDidDisappear(_ animated: Bool) {
    self.timer?.invalidate()
    self.timer = nil
    print("viewDidDisappear")
}

打印:SecondController 销毁

运行pushpop能够正常销毁,但此处会显着很刺眼,难道我们每一个定时器都要在这个地方处理吗。当然在iOS10系统中苹果提供了block,只需弱引用控制器即可。下面来感受一下(此处没有引用当前控制器):

self.timer = Timer.init(timeInterval: 1, repeats: true, block: {(timer) in
    print("timer")
})
RunLoop.current.add(self.timer!, forMode: .common)

打印:
SecondController 销毁
timer

我们可以看到,当前控制器被销毁了,但是定时器好像并没有停止打印,因此这里定时器并没有销毁,其实定时器是被RunLoop所持有,为解决这一问题,我们还是需要像上面那样使定时器失效并置空才能解决。

以上方法都不便于我们对定时器的管理,而我们理想中的定时器需要跟随引用者的释放而释放,我们只负责创建和处理定时器事件。那么我们就引入中介者,把timer的失效和置空交给中介者解决就行。

中介者实现

首先中介者要知道外界的方法选择器便于调用,再者为打破self->timer->self的循环引用,中介者内部对self做弱引用。声明属性如下:

weak var target: NSObjectProtocol?
var sel: Selector?
var timer: Timer? = nil
  • 管理外界对象,外界选择器,及内部的timer定时器

仿造Timer定时器方法,对外设置定时器方法接受外部调用对象即选择器:

func hb_scheduledTimer(timeInterval ti: TimeInterval, target aTarget: Any, selector aSelector: Selector, userInfo: Any?, repeats yesOrNo: Bool){
    self.timer = Timer.init(timeInterval: ti, target: self, selector: #selector(hb_timerFire), userInfo: userInfo, repeats: yesOrNo)
    RunLoop.current.add(self.timer!, forMode: .common)
    //此处对外self是弱引用
    self.target = (aTarget as! NSObjectProtocol)
    self.sel = aSelector
}
  • 内部定义定时器,设定target为当前中介者类
  • 设定定时器触发方法为中介者类的方法
  • 定时器加入到RunLoop
  • 保存外界目标对象及选择器

重点在中介者定时器触发方法中:

@objc fileprivate func hb_timerFire(){
    if self.target != nil{
        self.target!.perform(self.sel)
    }else{
        self.timer?.invalidate()
        self.timer = nil
    }
}
deinit {
    print("\(self.classForCoder) 走了 ")
}
  • 有目标对象,通过perform调用目标对象的方法
  • 没有目标对象,即清除定时器,解除RunLoop对定时器的引用

以上中介者已构建完成,下面调用测试一下:

class SecondController: UIViewController{
    let proxy = TimerProxy()

    override func viewDidLoad() {
        super.viewDidLoad()
        self.view.backgroundColor = .white
        self.proxy.hb_scheduledTimer(timeInterval: 1, target: self, selector: #selector(timerFire), userInfo: nil, repeats: true)
    }
    @objc func timerFire() {
        print("timer")
    }
    deinit {
        print("\(self.classForCoder) 销毁")
    }
}

pushpop页面打印如下:

timer
timer
SecondController 销毁
TimerProxy 走了 
  • 所有对象均被销毁,由于中介者内部对当前self是弱引用,所以当前控制器能够正常销毁
  • 当前控制器销毁后,proxy内部弱引用对象target也会被销毁
  • proxy内部定时器执行判断target时,发现targetnil即释放了定时器
  • proxy不持有timertimer也不持有proxy,即proxy会被销毁

RxSwift定时器

let timer = Observable<Int>.interval(1, scheduler: MainScheduler.instance)
timer.subscribe(onNext: { (time) in
    print(time)
}).disposed(by: disposeBag)

以上序列其实就是一个中介者,在RxSwift中管理了Timer对象,在当前对象销毁时清除垃圾袋并销毁定时器对象。

以上就是中介者对象的作用,直接使用,不用再对项目中定时器做释放操作。中介者其实就是封装复杂繁琐的操作,和不便于管理的业务,简化操作流程,让开发变得更简洁更高效。

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