iOS技术程序员

iOS消息分发中心的实现

2017-06-15  本文已影响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代码

更多的问题

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

上一篇下一篇

猜你喜欢

热点阅读