美文网首页iOS技术程序员
iOS消息分发中心的实现

iOS消息分发中心的实现

作者: 星___尘 | 来源:发表于2017-06-15 18:34 被阅读114次

后端的启发 && 前端的尴尬

最近一直在看React Native的一些相关设计,对其Redux的设计模式很感兴趣。Redux其实是一种响应式的设计,跟移动端中的MVVM有点类似,都是基于对状态的监听和绑定。当然,Redux跟MVVM还是有很大区别的,Redux的数据流是单向的,MVVM的不是。然而,无论是Redux还是MVVM,都离不开模块间的消息传递,无论是传递数据还是传递变化状态。对后端有一定了解的都知道,后端架构非常复杂,其中就包含了消息分发的功能。

随着移动端项目规模越来越大,模块间状态管理越来越复杂,各个组件间通讯成本越来越高,如果还是采用传统的Delegate,target-action,Notification,KVO来进行状态管理,那么会使得状态管理非常离散,到处都是Delegate代码。而对于全局状态的管理,Delegate就显得力不从心了。所以,为了解决越来越高的组件间通讯成本,需要引入一种类似于后端架构中的消息分发器,用来做消息转发的中介者,而不是组件和组件间的直接通讯。简单说就是组件间通讯的统一管理。

怎么设计

在iOS中,如何设计一个消息分发中心呢?首先要思考以下几个问题:

  • 一个消息体由什么组成
  • 使用什么方式进行发送
  • 如何对消息进行订阅
  • 分发中心如何分发消息
  • 是否需要先注册消息才能发送
  • 消息太多是否会阻塞
  • 如何减少对代码的入侵

Dispatch Cener 的设计

一个消息体,除了要包含消息内容外,还需要一个消息标识,作为消息的唯一标识,区分不同消息体。

YKMessage.h


@interface YKMessage : NSObject

/**
 消息唯一标识
 */
@property (nonatomic, readonly) NSString *identify;

/**
 消息内容
 */
@property (nonatomic, readonly) id context;

/**
 消息初始化函数

 @param identify 消息唯一标识
 @param context 内容
 @return 消息
 */
- (instancetype)initWithIdentify: (NSString *)identify context: (id)context;

YKMessage.m


@interface YKMessage()<NSCopying>

@property (nonatomic, readwrite) NSString *identify;

@property (nonatomic, readwrite) id context;

@end

@implementation YKMessage

- (instancetype)initWithIdentify: (NSString *)identify context: (id)context {
    self = [super init];
    if (self) {
        _identify = [identify copy];
        _context = [context copy];
    }
    return self;
}


- (id)copyWithZone:(NSZone *)zone {
    YKMessage *message = [[[self class]allocWithZone:zone]init];
    message.identify = self.identify;
    message.context = self.context;
    return message;
}

消息的发送:首先构造消息体,然后通过分发中心进行发送


    YKMessage *message = [[YKMessage alloc]initWithIdentify:@"test" context:@[@"1",@"2"]];
    [[YKDispatchCenter shared]dispatchMessage:message];

在需要接收消息的地方订阅消息:


    [[YKDispatchCenter shared]subscribeWithBinder:self messageIdentify:@"test" handler:^(YKMessage *message, id ext) {
        NSLog(@"dd");
    }];


这里有人会好奇为什么要加入binder这个参数,绑定self。我解析一下:
每一个消息的订阅者,都有自己的生命周期和作用域,当这个消息订阅者被释放后,基于它的消息回调也应该被释放掉,不应再被执行。因此回调是否被执行,就要看绑定者是否被释放了。

那传入binder后,分发中心怎么知道binder是否已经被释放了?

这里就要说一下一个弱应用可变数组NSPointerArray

平时使用NSMutableArrayNSArray的时候,其数组元素是不能为一个空值的,这是因为当数组元素被add进去的时候,该元素就被数组持有,内存引用计数会+1,而如果元素是空值的话,就会crash或报错。

但是,使用弱引用数组就不会有这种问题。弱引用数组添加元素的时候,会对元素进行一次弱引用,不会持有该元素,所以不会使元素的内存引用发生变化,因此即使add进一个空值,也不会crash或报错。

所以binder不会被消息分发中心持有,当binder被回收后,消息中心持有的弱引用数组中的binder弱引用也会变成空值,在执行回调前就可以通过这个来判断回调是否应该被执行了。

回到第四个问题:分发中心如何分发消息?

- (void)dispatchMessage: (YKMessage *)message {
    if (message == nil) {
        return;
    }
    if (![self.registerDictionary.allKeys containsObject:message.identify]) {
        return;
    }
    [self dealMessage:[message copy]];
}

- (void)dealMessage: (YKMessage *)message {
    // 实现异步发送通知
    dispatch_async(self.serialQueue, ^{
        [[NSNotificationCenter defaultCenter]postNotificationName:message.identify object:message];
    });
}

- (BOOL)registerMessageWithIdentify: (NSString *)messageIdentify {
    if ([self.registerDictionary.allKeys containsObject:messageIdentify]) {
        return NO;
    }
    [self.registerDictionary setValue:@"" forKey:messageIdentify];
    [[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(observerHandler:) name:messageIdentify object:nil];
    return YES;
}

- (BOOL)unRegisterMessageWithIdentify: (NSString *)messageIdentify {
    if (![self.registerDictionary.allKeys containsObject:messageIdentify]) {
        return NO;
    }
    [self.registerDictionary removeObjectForKey:messageIdentify];
    [self.actionDictionary removeObjectForKey:messageIdentify];
    [[NSNotificationCenter defaultCenter]removeObserver:self name:messageIdentify object:nil];
    return YES;
}

消息分发中心是基于通知来实现,在注册的时候将通知的发送者和接收者都绑定到自身上。通过发送通知和接收通知,来实现消息分发。

那为什么需要先注册消息才能发送呢?

首先,消息不是随便发就能发的。例如支付模块中,支付成功的消息必须是在支付模块中注册后才能发送,不能随便哪个模块就能直接发送支付成功的消息。在支付模块加载后注册支付成功的消息,在支付模块卸载后反注册支付成功消息,这样就能够控制消息发送的权限了。

其次,性能问题。有注册就有反注册,通过反注册销毁不需要维护的消息列表和通知观察者,减少性能消耗。

再次,业务问题。比如在某种情况下,不再需要某个消息了,所有这个消息的回调都不需要了。这时,通过反注册,就可以做到。

那消息太多是否会阻塞?

NSNotificationCenter在主线程中是同步的,当通知产生时,通知中心会一直等待所有观察者都收到且处理通知完毕后,才会返回发送通知的地方继续执行后面的代码。通常来说,如果消息太多,NSNotificationCenter会变慢。然而,这里通过创建一个serialQueue串行队列,并将消息的发送和接收放到这队列中执行,从而避免主队列的阻塞等待。


- (void)dealMessage: (YKMessage *)message {
    // 实现异步发送通知
    dispatch_async(self.serialQueue, ^{
        [[NSNotificationCenter defaultCenter]postNotificationName:message.identify object:message];
    });
}

- (void)observerHandler: (NSNotification *)notification {
    // 实现异步接收通知
    dispatch_async(self.serialQueue, ^{
        YKMessage *object = (YKMessage *)notification.object;
        if (object != nil) {
            NSString *messageIdentify = object.identify;
            [self actionAndCleanWithMessageIdentify:messageIdentify message:object doHandler:YES];
        }
    });
}

如果消息实在太多,还是会对性能有一定影响,但是这里对发送和接收通知进行异步操作,不会阻塞主线程。

那如何减少对代码的入侵?

// 订阅
    [[YKDispatchCenter shared]subscribeWithBinder:self messageIdentify:@"test" handler:^(YKMessage *message, id ext) {
        NSLog(@"dd");
    }];
    
// 发送
    YKMessage *message = [[YKMessage alloc]initWithIdentify:@"test" context:@[@"1",@"2"]];
    [[YKDispatchCenter shared]dispatchMessage:message];

简洁的API设计,简单的使用,是减少入侵和耦合的最好方式。

代码

项目代码
Demo代码

更多的问题

然而,这个消息分发中心并不完善,还有不少其他问题需要考虑:

  • 如何做消息优先级区分
  • 消息发送失败怎么办,是否支持重发
  • ......

相关文章

  • iOS消息分发中心的实现

    后端的启发 && 前端的尴尬 最近一直在看React Native的一些相关设计,对其Redux的设计模式很感兴趣...

  • iOS 使用 NSProxy 实现消息分发器

      最近准备进一步重构某几个页面,从结构上讲用的是 MVVM,较为清晰明了,同时也不至于所有代码都集中在 UIVi...

  • OTA方式来分发iOS的应用

    OTA分发的实现方式 针对iOS应用分发,需要在服务器里上包括三个文件,来实现OTA方式的分发 1.ipa文件,也...

  • 消息中心(实现)

    前言 在上一章主要梳理了信息中心的运行机制,了解了服务端是如何把更新的信息主动推送给客户端的,接下来我会介绍下消息...

  • iOS 方法交换(methodSwizzling)

    iOS运行中runtime中采用消息分发机制,允许开发者对其进行方法交换,替换某个类的方法实现或者在执行方法前添加...

  • NSNotification通知的使用和多线程

    通知的使用 NSNotificationCenter通知中心是iOS程序内部的一种消息广播的实现机制,可以在不同对...

  • iOS NSNotificationCenter Appdele

    通知的使用 NSNotificationCenter通知中心是iOS程序内部的一种消息广播的实现机制,可以在不同对...

  • NSNotificationCenter使用总结

    通知中心对于iOS开发者最熟悉不过了,它实现了一对多的消息传递,可以实现跨页面传递。NSNotificationC...

  • iOS打包自动化实践(四)

    上一篇:iOS打包自动化实践(三) 打包到分发全程自动化的闭环。 本篇文章介绍了iOS包自动化分发的实现。 效果 ...

  • iOS 通知 与 通知中心

    iOS 通知中心:自己实现了一套消息机制,可以跨页面调用 类似Unity的SendMessage,是订阅、发布者模...

网友评论

本文标题:iOS消息分发中心的实现

本文链接:https://www.haomeiwen.com/subject/sbjrfxtx.html