在以前的文章iOS奇思妙想之使用block替代通知(一) )中,本身实现了通知功能。在以前的实现中,使用了Runtime
进行动态绑定,虽然达到了最后的效果,可是也会增长耦合。 #解决耦合 既然动态绑定属性会增长耦合,那么咱们能够考虑不让观察者动态绑定属性来实现。这里参考以前NSMapTable
,将对应属性存入,在调用的时候再根据不一样的Key动态获取。git
根据不一样的观察者和监听类型动态生成Key,使用NSMapTable
将对应Key和观察者设置的属性储存。github
+ (void)addObserver:(id)observer type:(CLActionType)type mainThread:(BOOL)mainThread actionBlock:(void(^)(id observer, NSDictionary *dictionary))actionBlock {
dispatch_semaphore_wait([CLActionManager sharedManager].semaphore, DISPATCH_TIME_FOREVER);
NSString *key = [NSString stringWithFormat:@"%@-%@",[NSString stringWithFormat:@"%p",observer], [[self keyWithActionType:type] stringByAppendingString:@"-1"]];
NSString *actionBlockKey = [key stringByAppendingString:@"-CLActionBlock-1"];
NSString *actionMainThreadKey = [key stringByAppendingString:@"-CLActionMainThread-1"];
NSMutableDictionary *blockDictionary = [[CLActionManager sharedManager].blockDictionaryMapTable objectForKey:observer];
if (!blockDictionary) {
blockDictionary = [NSMutableDictionary dictionary];
}
[blockDictionary setObject:actionBlock forKey:actionBlockKey];
NSMutableDictionary *mainThreadDictionary = [[CLActionManager sharedManager].mainThreadDictionaryMapTable objectForKey:observer];
if (!mainThreadDictionary) {
mainThreadDictionary = [NSMutableDictionary dictionary];
}
[mainThreadDictionary setObject:[NSNumber numberWithBool:mainThread] forKey:actionMainThreadKey];
[[CLActionManager sharedManager].observerMapTable setObject:observer forKey:key];
[[CLActionManager sharedManager].blockDictionaryMapTable setObject:blockDictionary forKey:observer];
[[CLActionManager sharedManager].mainThreadDictionaryMapTable setObject:mainThreadDictionary forKey:observer];
dispatch_semaphore_signal([CLActionManager sharedManager].semaphore);
}
复制代码
根据观察者和监听类型动态生成Key,而后找出对应观察者的属性,而后一一调用。bash
+ (void)postType:(CLActionType)type dictionary:(NSDictionary *)dictionary {
dispatch_semaphore_wait([CLActionManager sharedManager].semaphore, DISPATCH_TIME_FOREVER);
NSArray<NSString *> *keyArray = [[[CLActionManager sharedManager].observerMapTable keyEnumerator] allObjects];
NSString *identifier = [[self keyWithActionType:type] stringByAppendingString:@"-1"];
NSPredicate *predicate = [NSPredicate predicateWithFormat:@"SELF ENDSWITH %@",identifier];
NSArray<NSString *> *array = [keyArray filteredArrayUsingPredicate:predicate];
for (NSString *key in array) {
NSString *actionBlockKey = [key stringByAppendingString:@"-CLActionBlock-1"];
NSString *actionMainThreadKey = [key stringByAppendingString:@"-CLActionMainThread-1"];
id observer = [[CLActionManager sharedManager].observerMapTable objectForKey:key];
NSMutableDictionary *blockDictionary = [[CLActionManager sharedManager].blockDictionaryMapTable objectForKey:observer];
NSMutableDictionary *mainThreadDictionary = [[CLActionManager sharedManager].mainThreadDictionaryMapTable objectForKey:observer];
void(^block)(id observer, NSDictionary *dictionary) = [blockDictionary objectForKey:actionBlockKey];
BOOL mainThread = [[mainThreadDictionary objectForKey:actionMainThreadKey] boolValue];
if (block) {
if (mainThread) {
dispatch_async(dispatch_get_main_queue(), ^{
block(observer, dictionary);
});
}else {
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
block(observer, dictionary);
});
}
}
}
dispatch_semaphore_signal([CLActionManager sharedManager].semaphore);
}
复制代码
这里为了使用方便,采起了Block回调的方式来监听,可是咱们也能够考虑协议分发的方式来实现,每一个观察者遵照Action协议,发送通知的时候咱们就能够找出对应观察者,调用协议中的方法来达到事件分发的目的。async
以上是对以前的代码进行改进,但愿可以给你们帮助,demo地址--->>CLActionManageride