依賴注入(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
則可以輕鬆實現前後端分離。
目前的缺點也很明顯,不能告訴開發者哪些屬性是必需依賴,另外還不能支援非物件屬性的賦值,希望拋磚引玉,大家來改進這段程式碼。