iOS實現依賴注入

秋刀生魚片發表於2016-03-22

依賴注入(Dependency Injection)這個詞,源於java,但在Cocoa框架中也是十分常見的。
舉例來說:
UIView的初始化方法initWithFrame

- (id)initWithFrame:(CGRect)frame NS_DESIGNATED_INITIALIZER;

這裡的frame傳入值,就是所謂的依賴(Dependency),這個View例項化是根據frame注入實現的。
但這種用法有很大的侷限性

  1. 我們不知道究竟依賴注入的屬性有哪些

  2. 不可能無限加長方法長度來滿足更多的依賴屬性

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

相關文章