相信我們開發(fā)的項(xiàng)目中郑象,只要涉及到網(wǎng)絡(luò)交互铝量,都會遇到一個(gè)再普遍的不過的需求,那就是出于用戶體驗(yàn)的需要银亲,在請求開始的時(shí)候顯示加載頁
慢叨,請求到空數(shù)據(jù)的時(shí)候顯示空內(nèi)容頁
,以及請求出錯(cuò)的時(shí)候顯示的錯(cuò)誤或重試頁
务蝠。這三類頁面在一個(gè)項(xiàng)目中通常是一致的(至多會有圖標(biāo)和文案的變化)拍谐,但卻要求可能在每一個(gè)涉及網(wǎng)絡(luò)請求的頁面呈現(xiàn)。
(馏段;′⌒`)
如果沒有大局觀赠尾,一開始接到需求就開始在某ViewController里面添加幾個(gè)View用來展現(xiàn)。
舉個(gè)例子毅弧,如果要添加一個(gè)loading view,
@property (strong, nonatomic) UIView *loadingView;
@property (strong, nonatomic) UIImageView *loadingImageView;
然后增加方法当窗,視圖的初始化和配置就省略了
/**
* 加載視圖
*/
- (void)startLoading{
[self.view addSubview:self.loadingView];
[self.loadingImageView startAnimating];
}
/**
* 停止加載并消失
*/
- (void)stopLoading{
[self.loadingImageView stopAnimating];
[self.loadingView removeFromSuperview];
}
OK,這樣做實(shí)現(xiàn)上沒問題,但是遇到下一個(gè)需要展示這些頁面的ViewController患久,只能使用copy&paste大法另萤,把property和方法實(shí)現(xiàn)都搬到另一個(gè)ViewController,倘若有10個(gè)以上的頁面巫员,再加上萬一需要修改頁面的視圖結(jié)構(gòu)庶香,你就會深刻的體會到
總所周知有個(gè)大原則叫做Don't repeat yourself。再運(yùn)用上我們不為什么就很熟練的面向?qū)ο笏季S简识,自然而然可以想到赶掖,使用繼承大法。
╮(╯_╰)╭
實(shí)現(xiàn)方法很簡單
首先創(chuàng)建一個(gè)BaseViewController七扰,將幾個(gè)property轉(zhuǎn)移過來奢赂,并且展現(xiàn)方法也照搬,在.h文件暴露出來颈走。
然后把所有用到的Controller都繼承自BaseViewController膳灶,調(diào)用的地方可以保持不變,會自動調(diào)用父類的方法立由。
但是這樣做還是不夠好轧钓,因?yàn)檫@需要我們把所有的Viewcontroller的頭文件都改一遍,引入BaseViewController并集成锐膜,就是所謂這是帶有侵入性
的毕箍。更致命的是,如果你的ViewController本身集成了另外的BaseController枣耀,由于Objective-C不支持多繼承霉晕,你只能去修改另一個(gè)BaseController……有點(diǎn)悲傷庭再。
╭(′▽`)╯
通過標(biāo)題的劇透,我們知道最后的實(shí)現(xiàn)跟runtime有關(guān)牺堰,那么主角也該出場了拄轻。
其實(shí)就是使用Category + runtime的對象關(guān)聯(lián)。在上面的方案中伟葫,解決集成BaseViewController的侵入性的方案就是使用category為UIViewController添加方法恨搓,但是category是不能直接使用property保存私有變量的,于是引入runtime的AssociatedObject系列方法筏养,可以動態(tài)為對象添加成員變量斧抱,這幾乎是runtime最基礎(chǔ)的應(yīng)用。
非常簡單的渐溶,只用到兩個(gè)方法辉浦,其實(shí)就是一個(gè)Setter和Getter
void objc_setAssociatedObject(id object, const void *key, id value, objc_AssociationPolicy policy) //為object添加一個(gè)value,名字是key
id objc_getAssociatedObject(id object, const void *key) //從object中取一個(gè)名字是key的value
如果想深入探究一下茎辐,有人已經(jīng)寫得挺全面了宪郊,可以點(diǎn)擊這篇文章
在本例中,用法大概是這樣
#pragma mark - Getter
- (UIView *)loadingView{
UIView *loadingView = objc_getAssociatedObject(self, &PresnterLoadingViewKey);
if (!loadingView) {
loadingView = [[UIView alloc] initWithFrame:self.view.bounds];
objc_setAssociatedObject(self, &PresnterLoadingViewKey, loadingView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
loadingView.backgroundColor = [UIColor whiteColor];
[loadingView addSubview:self.loadingImageView];
}
return loadingView;
}
- (UIImageView *)loadingImageView{
UIImageView *imageView = objc_getAssociatedObject(self, &PresnterLoadingImageViewKey);
if (!imageView) {
imageView = [[UIImageView alloc] initWithFrame:
CGRectMake(self.view.bounds.size.width / 2 - 100, self.view.bounds.size.height/2 - 80, 200, 150)];
NSMutableArray *tmpArr = [NSMutableArray array];
for (int i = 0; i <= 80; i++) {
UIImage *image = [UIImage imageNamed:[NSString stringWithFormat:@"01-progress00%02d.jpg",i]];
[tmpArr addObject:image];
}
[imageView setAnimationImages:[NSArray arrayWithArray:tmpArr]];
imageView.animationDuration = 2.0;
objc_setAssociatedObject(self, &PresnterLoadingImageViewKey, imageView, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
return imageView;
}
把實(shí)例化放進(jìn)Getter拖陆,這樣實(shí)現(xiàn)方法可以同上保持不變
/**
* 加載視圖
*/
- (void)startLoading{
[self.view addSubview:self.loadingView];
[self.loadingImageView startAnimating];
}
/**
* 停止加載并消失
*/
- (void)stopLoading{
[self.loadingImageView stopAnimating];
[self.loadingView removeFromSuperview];
}
最后只需要在你需要用到這些頁面的ViewController引入UIViewController+Presenter.h
然后展示就好
[self startLoading]; //加載完成后調(diào)用 [self stopLoading];
具體的代碼我寫了個(gè)demo放在github上弛槐,地址在這里。
另外實(shí)現(xiàn)了空白視圖和失敗重試視圖的功能依啰,跟loading頁大同小異乎串。
如此一來,這些公共頁面的展示邏輯基本被封裝進(jìn)了category中速警,而且當(dāng)我們需要修改展示的頁面時(shí)叹誉,也只需修改文件里面的實(shí)現(xiàn),然后暴露出方法闷旧,簡單高效桂对。