RxSwift中的五种Subject

Subject.png

Subject是一种桥梁和代理,在ReactiveX的一些实现中,它既可以当作observer也可以当做Observable。因为它是一个observer,所以它可以订阅一个或多个Observable,同时因为它是一个Observable,它可以传递它观察到的事件,重新发送他们,它也可以发送新的事件。

Subject订阅一个Observable时,它可以触发Observable发送事件(如果那是一个“cold” Observable,以为“cold” Observable会在被订阅之后开始发送信息)。这一特点可以让Subject成为一个“hot”Observable,这个“hot” Observable是一种原来“cold”Observable的变体。

AsyncSubject

一个AsyncSubject只在原始Observable完成后,发射来自原始Observable的最后一个值。(如果原始Observable没有发射任何值,AsyncObject也不发射任何值)它会把这最后一个值发射给任何后续的观察者。

S.AsyncSubject.png

然而,如果原始的Observable因为发生了错误而终止,AsyncSubject将不会发射任何数据,只是简单的向前传递这个错误通知。

S.AsyncSubject.e.png
      let subject = AsyncSubject<String>()
        
        subject.onNext("Episode1 updated")

        _ = subject.subscribe(onNext: {
            print("Sub1 - what happened: \($0)")
        })
        
        subject.onNext("Episode2 updated")
        let error = NSError.init(domain: "", code: 2, userInfo: ["213" :232])
//        subject.onError(error)
        subject.onCompleted()

PublishSubject

PublishSubject只会把在订阅发生的时间点之后来自原始Observable的数据发射给观察者。需要注意的是,PublishSubject可能会一创建完成就立刻开始发射数据(除非你可以阻止它发生),因此这里有一个风险:在Subject被创建后到有观察者订阅它之前这个时间段内,一个或多个数据可能会丢失。如果要确保来自原始Observable的所有数据都被分发,你需要这样做:或者使用Create创建那个Observable以便手动给它引入"冷"Observable的行为(当所有观察者都已经订阅时才开始发射数据),或者改用ReplaySubject

S.PublishSubject.png
       let subject = PublishSubject<String>()
        let sub1 = subject.subscribe(onNext: {
            print("Sub1 - what happened: \($0)")
        })
        subject.onNext("Episode1 updated")

        /*
         但是执行一下就会发现,控制台上不会显示任何订阅消息,也就是说sub1没有订阅到任何内容。这是因为PublishSubject执行的是“会员制”,它只会把最新的消息通知给消息发生之前的订阅者。
         */
        
        sub1.dispose()
        
        let sub2 = subject.subscribe(onNext: {
            print("Sub2 - what happened: \($0)")
        })
        
        subject.onNext("Episode2 updated")
        subject.onNext("Episode3 updated")
        
        sub2.dispose()

BehaviorSubject

当观察者订阅BehaviorSubject时,它开始发射原始Observable最近发射的数据(如果此时还没有收到任何数据,它会发射一个默认值),然后继续发射其它任何来自原始Observable的数据。

S.BehaviorSubject.png
       /*
         如果你希望Subject从“会员制”变成“试用制”,就需要使用BehaviorSubject。
         它和PublisherSubject唯一的区别,就是只要有人订阅,
         它就会向订阅者发送最新的一次事件作为“试用”。
         */
        let subject = BehaviorSubject<String>(value: "RxSwift step by step")
        
        _ = subject.subscribe(onNext: {
            print("Sub1 - what happened: \($0)")
        })
        
        subject.onNext("Episode1 updated")
        /*
            由于BehaviorSubject有了一个默认的事件,sub1订阅之后,
            就会陆续收到RxSwift step by step和Sub1 - what happened: Episode1 updated的消息了。
            此时,如果我们再添加一个新的订阅者:
         */
        _ = subject.subscribe(onNext: {
            print("Sub2 - what happened: \($0)")
        })
        /*
         此时,sub2就只能订阅到Sub2 - what happened: Episode1 updated消息了。
         */

ReplaySubject

ReplaySubject会发射所有来自原始Observable的数据给观察者,无论它们是何时订阅的。也有其它版本的ReplaySubject,在重放缓存增长到一定大小的时候或过了一段时间后会丢弃旧的数据(原始Observable发射的)。

如果你把ReplaySubject当作一个观察者使用,注意不要从多个线程中调用它的onNext方法(包括其它的on系列方法),这可能导致同时(非顺序)调用,这会违反Observable协议,给Subject的结果增加了不确定性。

S.ReplaySubject.png
       /*
         ReplaySubject的行为和BehaviorSubject类似,都会给订阅者发送历史消息。不同地方有两点:
         
         ReplaySubject没有默认消息,订阅空的ReplaySubject不会收到任何消息;
         ReplaySubject自带一个缓冲区,当有订阅者订阅的时候,它会向订阅者发送缓冲区内的所有消息;
         */
        
        let subject = ReplaySubject<String>.create(bufferSize: 3)
        
        _ = subject.subscribe(onNext: {
            print("Sub1 - what happened: \($0)")
        })
        
        subject.onNext("Episode1 updated")
        subject.onNext("Episode2 updated")
        subject.onNext("Episode3 updated")
        
        _ = subject.subscribe(onNext: {
            print("Sub2 - what happened: \($0)")
        })

        /*
         由于subject缓冲区的大小是3,它会自动给sub2发送最新的三次历史事件。在控制台中执行一下,就可以看到注释中的结果了。
         */

Variable

除了事件序列之外,在平时的编程中我们还经常需遇到一类场景,就是需要某个值是有“响应式”特性的,例如可以通过设置这个值来动态控制按钮是否禁用,是否显示某些内容等。为了方便这个操作,RxSwift还提供了一个特殊的subject,叫做Variable

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

推荐阅读更多精彩内容