iOS多代理实现
XMPP以及类似IM框架里通常会有这种需求:打开多个聊天窗口,和多个人聊天。然鹅框架底层消息转发管理器却只有一个。通常是这两个窗口都要收到消息回调,然后取自己有用的消息。。。
大概就这么个意思,我两年前用了下,具体也解释不清楚,欢迎指点交流,反正就是有一个需求需要多代理回调,这种IM框架通常有这种方法
[xmppRoom addDelegate:self delegateQueue:dispatch_get_main_queue()];
这里有几个疑问
- 为什么不用通知
- 很好奇为什么他们的代理可以一直add
解释
关于为什么不用通知,这里面有很多渊源。大概介绍下,就是代理用的协议仿佛是一个“接口文档”,别人点进去协议,大概就知道怎么用了。如果XMPP设计成通知,则必须写一份文档详细告诉别人,哪个通知的名字是干什么的,调用者好对应接收,不太好处理
第二个问题就是写这篇文章目的,讲解下怎么实现多代理,也算是自己的一个备份,万一遇到这种需求呢
开始
用数组接收,数组由管理类强引用,一般这种管理类会是单例类一直存在运行过程中,数组又强引用代理,最终会导致所有代理都没法释放
NSPointerArray
然后看到有些文章介绍到NSPointerArray,这个应该可以,好多文章都介绍了,我比较懒没有尝试
桥接类
当时不知道再哪搜的资料,提到用中间桥接类。感觉这个实现很巧妙,而且都是熟悉的API,我自己实现了一下,发现确实很完美的实现需求
DemoClass.h
#import <Foundation/Foundation.h>
@protocol DemoClassDelegate <NSObject>
- (void)testMethodInProtocol;
- (void)testMethodWithParameter:(NSString *)param;
@end
@interface DemoClass : NSObject
- (void)addDelegate:(id<DemoClassDelegate>)delegate;
@end
可以把这个DemoClass当成管理类,由他回调他所有的代理,我们会在一开始把它的实例保存在AppDelegate中或者或者其他便于找到的管理类
DemoClass.m
#import "DemoClass.h"
// 桥接类 弱引用delegate
@interface DemoClassBridge : NSObject
@property (nonatomic, weak) id<DemoClassDelegate> delegate;
@end
@implementation DemoClassBridge
- (void)dealloc
{
NSLog(@"%s", __func__);
}
@end
@interface DemoClass ()
@property (nonatomic, strong) NSMutableArray *delegatesContainer;
@end
@implementation DemoClass
- (instancetype)init
{
self = [super init];
if (self) {
// 此处模拟网络请求后或者其他需要回调时
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(10 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
[self startCallBack];
});
}
return self;
}
- (void)startCallBack
{
int random = arc4random()%100;
for (DemoClassBridge *obj in self.delegatesContainer) {
if ([obj.delegate respondsToSelector:@selector(testMethodInProtocol)]) {
[obj.delegate testMethodInProtocol];
}
if ([obj.delegate respondsToSelector:@selector(testMethodWithParameter:)]) {
[obj.delegate testMethodWithParameter:[NSString stringWithFormat:@"传了一个参数%d",random]];
}
}
}
- (void)addDelegate:(id<DemoClassDelegate>)delegate
{
DemoClassBridge *bridge = [DemoClassBridge new];
bridge.delegate = delegate;
[self.delegatesContainer addObject:bridge];
}
- (NSMutableArray *)delegatesContainer
{
if (nil == _delegatesContainer) {
_delegatesContainer = [[NSMutableArray alloc]init];
}
return _delegatesContainer;
}
- (void)dealloc
{
NSLog(@"%s", __func__);
}
这样就OK了,将三个VC作为代理加进去,延迟10秒后都可以收到回调。
add进去作为代理期间,由桥接类弱引用该VC,如果这个VC,pop出去,那么就会直接释放,回调时,遍历数组桥接类对象的delegate属性为nil,就不会触发回调。
测试结果
所有的vc以及这个管理类自身,都可以正常释放。代理可以收到对应的回调,以及传过来的参数~
demo项目链接