iOS消息分发中心的实现

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

最近一直在看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代码

更多的问题

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

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

推荐阅读更多精彩内容