前两篇文章主要是对这篇文章的内容进行了一个铺垫,这里就一块儿来看下
CTMediator
的实现原理 ,CTMediator
是一个单例,主要是基于Mediator
模式和Target-Action
模式,中间采用了runtime
来完成调用swift
CTMediator
提供的API分别为:远程app调用入口
、本地组件调用入口
、释放某个target缓存
这里主要介绍 本地组件调用入口
也是咱们最经常使用的一个方法:缓存
- (id)performTarget:(NSString *)targetName action:(NSString *)actionNameparams:(NSDictionary *)params shouldCacheTarget:(BOOL)shouldCacheTarget;
复制代码
实现分析:bash
// 从 params 字典中 获取 swiftModuleName
NSString *swiftModuleName = params[kCTMediatorParamsKeySwiftTargetModuleName];
// generate target
NSString *targetClassString = nil;
if (swiftModuleName.length > 0) {
targetClassString = [NSString stringWithFormat:@"%@.Target_%@", swiftModuleName, targetName];
} else {
targetClassString = [NSString stringWithFormat:@"Target_%@", targetName];
}
// 根据 targetClassString 从 cachedTarget (缓存的Target)获取 target
NSObject *target = self.cachedTarget[targetClassString];
if (target == nil) {
// 未获取到 则经过NSClassFromString将字符串转为应的类
Class targetClass = NSClassFromString(targetClassString);
target = [[targetClass alloc] init];
}
// generate action
NSString *actionString = [NSString stringWithFormat:@"Action_%@:", actionName];
SEL action = NSSelectorFromString(actionString);
if (target == nil) {
// 这里是处理无响应请求的地方之一,这个demo作得比较简单,若是没有能够响应的target,就直接return了。实际开发过程当中是能够事先给一个固定的target专门用于在这个时候顶上,而后处理这种请求的
[self NoTargetActionResponseWithTargetString:targetClassString selectorString:actionString originParams:params];
return nil;
}
// 是否须要对 Target 进行缓存
if (shouldCacheTarget) {
// 将 Target 进行缓存
self.cachedTarget[targetClassString] = target;
}
// 判断target对象是否响应action,避免crash
if ([target respondsToSelector:action]) {
// 这里是处理有响应请求的地方
return [self safePerformAction:action target:target params:params];
} else {
// 这里是处理无响应请求的地方,若是无响应,则尝试调用对应target的notFound方法统一处理
SEL action = NSSelectorFromString(@"notFound:");
if ([target respondsToSelector:action]) {
return [self safePerformAction:action target:target params:params];
} else {
// 这里也是处理无响应请求的地方,在notFound都没有的时候,这个demo是直接return了。实际开发过程当中,能够用前面提到的固定的target顶上的。
[self NoTargetActionResponseWithTargetString:targetClassString selectorString:actionString originParams:params];
// 删除缓存的无用 Target
[self.cachedTarget removeObjectForKey:targetClassString];
return nil;
}
}
复制代码
处理有响应请求的地方会调用 - (id)safePerformAction:(SEL)action target:(NSObject *)target params:(NSDictionary *)params
方法app
// NSMethodSignature 记录着某个方法的返回值类型信息以及参数类型信息。用于转发消息接收者没法响应的消息
NSMethodSignature* methodSig = [target methodSignatureForSelector:action];
if(methodSig == nil) {
return nil;
}
// 获取返回类型
const char* retType = [methodSig methodReturnType];
// 判断返回值 类型
if (strcmp(retType, @encode(void)) == 0) {
// 用来包装方法和对应的对象,它能够存储方法的名称,对应的对象,对应的参数
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSig];
[invocation setArgument:¶ms atIndex:2];
[invocation setSelector:action];
[invocation setTarget:target];
// 执行NSInvocation对象中指定对象的指定方法,而且传递指定的参数
[invocation invoke];
return nil;
}
if (strcmp(retType, @encode(NSInteger)) == 0) {
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSig];
[invocation setArgument:¶ms atIndex:2];
[invocation setSelector:action];
[invocation setTarget:target];
[invocation invoke];
NSInteger result = 0;
// 将返回数据拷贝到提供的缓存区(retLoc)内
[invocation getReturnValue:&result];
return @(result);
}
if (strcmp(retType, @encode(BOOL)) == 0) {
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSig];
[invocation setArgument:¶ms atIndex:2];
[invocation setSelector:action];
[invocation setTarget:target];
[invocation invoke];
BOOL result = 0;
[invocation getReturnValue:&result];
return @(result);
}
if (strcmp(retType, @encode(CGFloat)) == 0) {
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSig];
[invocation setArgument:¶ms atIndex:2];
[invocation setSelector:action];
[invocation setTarget:target];
[invocation invoke];
CGFloat result = 0;
[invocation getReturnValue:&result];
return @(result);
}
if (strcmp(retType, @encode(NSUInteger)) == 0) {
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSig];
[invocation setArgument:¶ms atIndex:2];
[invocation setSelector:action];
[invocation setTarget:target];
[invocation invoke];
NSUInteger result = 0;
[invocation getReturnValue:&result];
return @(result);
}
// 利用RunTime 向target对象传递消息,执行 target 中 action 的方法,传递参数 params
return [target performSelector:action withObject:params];
复制代码
总结: CTMediator
根据得到的target和action信息,经过objective-C的runtime转化生成target实例以及对应的action选择器,而后最终调用到目标业务提供的逻辑,完成需求。ui