ReactiveCocoa信号发送详解

简介

ReactiveCocoa 是一个重型的 FRP (Functional Reactive Programming 是一种响应变化的编程范式) 框架。内部使用了大量的block。FRP的核心就是信号。
RACSignal就是信号,是ReactiveCocoa中很重要的一个概念。RACSignal本体是RACStream。信号就是数据流,可以用来传递和绑定。

以下代码基于V2.5的ReactiveCocoa

创建RACsignal

不说废话,先来一张图

RACSignal.png
// 源码
+ (RACSignal *)createSignal:(RACDisposable * (^)(id<RACSubscriber> subscriber))didSubscribe {
    RACDynamicSignal *signal = [[self alloc] init];
    signal->_didSubscribe = [didSubscribe copy];
    return [signal setNameWithFormat:@"+createSignal:"];
}

通过RACDynamicSignal创建信号,此时传入一个block,这个block的参数是一个遵循RACSubscriber协议的一个变量,同时这个block的返回值是一个RACDisposable类型。

通过源码分析,看到RACDynamicSignal有一个属性didSubscribe存储了传进来的的block,这个属性将在之后订阅的时候使用。

这个RACSubscriber的协议,其中定义了几个方法

@protocol RACSubscriber <NSObject>
@required

// 发送next需要执行的参数
- (void)sendNext:(id)value;
// 发送错误
- (void)sendError:(NSError *)error;
// 发送成功
- (void)sendCompleted;
// 处理信号,是否释放取消订阅。
- (void)didSubscribeWithDisposable:(RACCompoundDisposable *)disposable;

@end

创建一个信号

RACSignal *aSignal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    [subscriber sendNext:@"a"];
    [subscriber sendCompleted];
    return [RACDisposable disposableWithBlock:^{
        
    }];
}];

订阅

一个信号通过调用subscribeNext创建一个subscriber进行subscription。

// RACSignal (Subscription) RACSignal.m
- (RACDisposable *)subscribeNext:(void (^)(id x))nextBlock {
    NSCParameterAssert(nextBlock != NULL);
    
    RACSubscriber *o = [RACSubscriber subscriberWithNext:nextBlock error:NULL completed:NULL];
    return [self subscribe:o];
}

// 当前self是RACDynamicSignal所以,使用RACDynamicSignal.m中的subscribe:方法。
// RACDynamicSignal.m
- (RACDisposable *)subscribe:(id<RACSubscriber>)subscriber {
    NSCParameterAssert(subscriber != nil);

    RACCompoundDisposable *disposable = [RACCompoundDisposable compoundDisposable];
    subscriber = [[RACPassthroughSubscriber alloc] initWithSubscriber:subscriber signal:self disposable:disposable];

    if (self.didSubscribe != NULL) {
        RACDisposable *schedulingDisposable = [RACScheduler.subscriptionScheduler schedule:^{
            RACDisposable *innerDisposable = self.didSubscribe(subscriber);
            [disposable addDisposable:innerDisposable];
        }];

        [disposable addDisposable:schedulingDisposable];
    }
    
    return disposable;
}
  1. 源码中创建了一个RACSubscriber来存储nextBlock、error、completed。然后订阅处理subscription。

  2. 这里有一个RACCompoundDisposable,这是一个RACDisposable,只不过RACCompoundDisposable可以存放多个RACDisposable 。当RACCompoundDisposable 执行dispose方法时,它所存放的disposable都会被释放。

  3. 使用RACPassthroughSubscriber将当前的订阅者进行转化,转化为另外一种形式的订阅者。这个订阅者中存储了当前的订阅者,信号、disposable。存储了一个信号的完整处理,并且这个订阅者同样遵循<RACSubscriber>协议。这里可以把RACPassthroughSubscriber当成是订阅者的装饰器(伪装器)。

  4. 使用RACPassthroughSubscriber的目的是将subscirber传递给另一个还没有disposed的subscriber。

    Passes through all events to another subscriber while not disposed.

  5. 当执行self.didSubscribe(subscriber)时siganle存储的block就会被执行。当sendNext:执行时,先执行[RACPassthroughSubscriber sendNext:],然后调用RACPassthroughSubscriber中的subscriber来执行sendNext:

    // 源码
        // 源码
    - (void)sendNext:(id)value {
       if (self.disposable.disposed) return;
    
       if (RACSIGNAL_NEXT_ENABLED()) {
          RACSIGNAL_NEXT(cleanedSignalDescription(self.signal), cleanedDTraceString(self.innerSubscriber.description), cleanedDTraceString([value description]));
       }
    
       [self.innerSubscriber sendNext:value];
    

}
继续执行addDisposable,此时会将RACCompoundDisposable```释放。

```objc
// 源码
- (void)addDisposable:(RACDisposable *)disposable {
NSCParameterAssert(disposable != self);
if (disposable == nil || disposable.disposed) return;

BOOL shouldDispose = NO;

OSSpinLockLock(&_spinLock);
{
    if (_disposed) {
        shouldDispose = YES;
    } else {
        #if RACCompoundDisposableInlineCount
        for (unsigned i = 0; i < RACCompoundDisposableInlineCount; i++) {
            if (_inlineDisposables[i] == nil) {
                _inlineDisposables[i] = disposable;
                goto foundSlot;
            }
        }
        #endif

        if (_disposables == NULL) _disposables = RACCreateDisposablesArray();
        CFArrayAppendValue(_disposables, (__bridge void *)disposable);

        if (RACCOMPOUNDDISPOSABLE_ADDED_ENABLED()) {
            RACCOMPOUNDDISPOSABLE_ADDED(self.description.UTF8String, disposable.description.UTF8String, CFArrayGetCount(_disposables) + RACCompoundDisposableInlineCount);
        }

    #if RACCompoundDisposableInlineCount
    foundSlot:;
    #endif
    }
}
OSSpinLockUnlock(&_spinLock);

// Performed outside of the lock in case the compound disposable is used
// recursively.
// 会在此处释放。
if (shouldDispose) [disposable dispose];

}


6. 源码中可以看出,订阅一个信号,返回的是一个RACDisposable,作为一个返回值返回到外部,我们可以在外部对其取消这个订阅。

###总结
```objc
// part 1.
RACSignal *aSignal = [RACSignal createSignal:^RACDisposable *(id<RACSubscriber> subscriber) {
    // 此处subscriber为转换后的subscriber
    // part 5.
   [subscriber sendNext:@"abc"];
   [subscriber sendCompleted];
    
   // part 6.
   return [RACDisposable disposableWithBlock:^{
       NSLog(@"disposable");
   }];
}];

// part 2.
RACDisposable *adisposable = [aSignal subscribeNext:^(id x) {
   NSLog(@"~~~~~~~~~~~  %@",x);
}];
  1. 调用createSignal:创建一个信号。存储当前的block到didSubscribe这个block中。
  2. 调用subscribeNext:订阅信号。创建一个subscriber来subscription。在subscriber中存储nextBlock,errorBlock,completeBlock三个block。
  3. 当前的订阅通过转换,成为RACPassthroughSubscriber,这个subscriber中有三个重要的属性分别是当前订阅的subscriber,当前的signal和RACCompoundDisposable。
  4. RACDynamicSignal执行didSubscribe(RACPassthroughSubscriber)这个block。执行RACPassthroughSubscriber中的sendNext:, sendError:, sendCompeleted
  5. RACPassthroughSubscriber中通过判断当前的disposable的状态来判断是否告诉subscriber调用相应的sendNext:等。
  6. 调用dispose方法,完成整个过程。

用一张图来来表示整个过程


ReactiveCocoa.jpg

引用

ReactiveCocoa gitHub

美团点评技术团队RACSignal的Subscription深入分析

iOS ReactiveCocoa详解

写在最后

第一篇ReactiveCocoa的文章,写的不好,如有写的不对的地方,欢迎指正,共同进步。谢谢!!!

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

推荐阅读更多精彩内容