iOS-创建自己的Signal工具类(一)

本文探讨了为何在工程中避免使用ReactiveCocoa,认为其工具包过大、调试困难、信号过多且资源占用大。然后介绍了如何创建自己的Signal工具类,包括理解订阅者和发布者概念,以及订阅和发布信号的流程。通过示例代码展示了如何实现信号的订阅和发送,解释了Block在其中的作用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

下载DEMO

最近看一些招聘信息的时候,尽然有人提到熟悉ReactiveCocoa。并且还是所谓的加分项。对于这个工具,熟练使用就好。一般工程不建议使用。

个人暂时觉得有一下几点:(以后待补充–不喜欢就别喷)
1:工具包太大
2:出问题了,调试非常不容易。
3:信号漫天飞。
4:工程需要用到的功能,可能就需要ReactiveCocoa他的一点点功能而已。资源占用太大

好了废话不多说。直接进入主题。

不用去罗列图片什么的,咱们直接进入代码看。

  1. 先要弄明白两个名词
    1:订阅者-接受信号,接受发布者发出的信号内容
    2:发布者-发布信号,订阅者发送数据源
  2. 使用过程
    1:订阅者就用subscribeNext:(void (^)(id x))nextBlock
    2: 发布信号sendNext:(id)value
  3. 发布到订阅流程
    首先订阅信号
    在订阅信号的时候,主要通过订阅RACBaseProctal这个协议。

顺便当前的对象也接受了这个协议,就要实现这个接口的方法。
编写自己的信号,我就写了其中的一个方法

B1C0202B-0AE7-469A-AD25-0F8F2873365A.png

在订阅信号的时候,系统做了一个假的堆栈。其实就是用一个数组,去装填接受了订阅RACBaseProctal的对象。
于是乎看下面的代码

// 组装对象
- (NSMutableArray * )subscribeNext:(void (^)(id x))nextBlock {
    NSCParameterAssert(nextBlock != NULL);
    RACBaseProctal * Proctal = [RACBaseProctal subscriberWithNext:nextBlock];
    return [self subscribe:Proctal];
}
// 向数组中装填接受了RACBaseProctal的对象
- (NSMutableArray *)subscribe:(id<RACBaseProctal>)subscriber {
    NSCParameterAssert(subscriber != nil);
    NSMutableArray *subscribers = self.subscribers;
    @synchronized (subscribers) {
        [subscribers addObject:subscriber];
    }
    
    @synchronized (subscribers) {
        // 根据条件用来获取一个NSUIndex 对象,主要是根据条件进行数据遍历使用
        // 这里添加了一个参数,用来表示遍历是从前向后遍历还是从后遍历。
        // NSInteger index =  [subscribers indexOfObjectWithOptions:(NSEnumerationReverse) passingTest:^BOOL(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
       //    return obj == subscriber;
       // }];
       // if (index != NSNotFound) [subscribers removeObjectAtIndex:index];
    }
    return subscribers;
}

NOTICE:在这个过程中,组装了一个装有RACBaseProctal的对象
的数组,并且还把(void (^)(id x))nextBlock一块组装。

重点来了!!!!

注意这个nextBlock就是我们要使用的发送信号的Block。

4.发布流程
看看发布信号的代码

- (void)enumerateSubscribersUsingBlock:(void (^)(id<RACBaseProctal> subscriber))block {
    NSArray *subscribers;
    @synchronized (self.subscribers) {
        subscribers = [self.subscribers copy];
    }
    for (id<RACBaseProctal> subscriber in subscribers) {
        block(subscriber);
    }
}

#pragma mark - RACBaseProctalDelegate
- (void)sendNext:(id)value {
    [self enumerateSubscribersUsingBlock:^(id<RACBaseProctal> subscriber) {
        [subscriber sendNext:value];
    }];
}

这里就是要循环遍历在订阅信号的时候组建的数组了。换一句话说就是,你有多少个订阅者,就会对应生产多大的数组。一个发布者就可以对应多个订阅者了。这里是精髓

发布就很简单了就是一个Block。
但是这个Block和我们订阅信号的Block对比看一下。
我们发信信号的Block

-(void)sendNext:(id)value {
    @synchronized (self) {
        void (^nextBlock)(id) = [self.next copy];
        if (nextBlock == nil) return;
        nextBlock(value);
    }
}

订阅发布信号的Block

+ (instancetype)subscriberWithNext:(void (^)(id x))next {
    RACBaseProctal *protocal = [[self alloc] init];
    protocal->_next = [next copy];
    return protocal;
}

这所有的信息都在这个RACBaseProctal接口协议中完成。对比我们可知。
这么一系列的转换,其实就是在搞同一个Block。发布信号就执行这个Block,订阅信号就输出这个。

于是乎,之乎者也。

有些地方说的不到位,还请各位看官指正。。。

  • 欢迎各位一块学习,提高逼格!
  • 也可以添加洲洲哥的微信公众号

可以来微信公众号(洲洲哥)后台给我们留言。 快来扫码关注我们吧!

公众号二维码

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值