依赖注入(Dependency Injection)
这个词,源于java,但在Cocoa框架中也是十分常见的。
举例来说:
UIView的初始化方法initWithFrame
- (id)initWithFrame:(CGRect)frame NS_DESIGNATED_INITIALIZER;
这里的frame传入值,就是所谓的依赖(Dependency)
,这个View实例化是根据frame注入实现的。
但这种用法有很大的局限性
我们不知道究竟依赖注入的属性有哪些
不可能无限加长方法长度来满足更多的依赖属性
所以我们准备采用字典容器对NSObject类进行依赖注入扩展。
给NSObject类添加一个Category
@interface NSObject (XXXDependencyInjection)
- (nullable id)initWithParams:(nonnull NSDictionary *)params;
- (void)injection:(nonnull NSDictionary*)params;
@end
实现注入方法
- (id)initWithParams:(NSDictionary *)params
{
self = [self init];
if (self) {
[self injection:params];
}
return self;
}
- (void)injection:(NSDictionary*)params
{
[params.allKeys enumerateObjectsUsingBlock:^(id _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
SEL selector = NSSelectorFromString([NSString stringWithFormat:@"set%@%@:",[[obj substringToIndex:1] uppercaseString],[obj substringFromIndex:1]]);
id value = [params objectForKey:obj];
if ([self respondsToSelector:selector]) {
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
[self performSelector:selector withObject:value];
#pragma clang diagnostic pop
}
else
{
@try {
[self setValue:value forKeyPath:obj];
}
@catch (NSException *exception) {
NSLog(@"%@",exception);
[exception raise];
}
@finally {
}
}
}];
}
解释
我们将需要注入的属性,封装到一个字典里,例如:
UIViewController* controller = [[UIViewController alloc] initWithParams:@{
@"title":@"测试",
@"view.backgroundColor":[UIColor whiteColor]
}];
我们给这个VC注入了两个属性,一个是其title,一个是其View的backgroundColor属性。
字典传入以后,我们读区params.allKeys
进行遍历,拼装set+参数名的selector,这里用的是NSSelectorFromString方法:
SEL selector = NSSelectorFromString([NSString stringWithFormat:@"set%@%@:",[[obj substringToIndex:1] uppercaseString],[obj substringFromIndex:1]]);
然后我们判断实例是否可以响应这个set方法,如果可以,则给其赋值。
if ([self respondsToSelector:selector]) {
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Warc-performSelector-leaks"
[self performSelector:selector withObject:value];
#pragma clang diagnostic pop
}
这里的三行clang宏是为了消除编译器的内存泄漏警告,这里因为我们进行了验证,所以不会出现leak。
KVC实现跨实例赋值
我们注意到上例中还有一句给VC的View改变背景颜色
@"view.backgroundColor":[UIColor whiteColor]
这里就用到了KVC的点语法特性,在我们判断到实例不能响应 if ([self respondsToSelector:selector])
的时候,通过点语法,进行赋值
@try {
[self setValue:value forKeyPath:obj];
}
@catch (NSException *exception) {
NSLog(@"%@",exception);
[exception raise];
}
@finally {
}
这里添加了异常捕获,因为点语法对属性名称拼写要求是全匹配,否则抛异常,所以要注意。
优缺点
这样改造过的init方法,优点非常明显,就是绑定更加集中便捷,如果使用的是storyboard
则可以轻松实现前后端分离。
目前的缺点也很明显,不能告诉开发者哪些属性是必需依赖,另外还不能支持非对象属性的赋值,希望抛砖引玉,大家来改进这段代码。