代理設計模式對於iOS開發的人來講確定很熟悉了,代理delegate就是委託另外一個對象來幫忙完成一件事情,爲何要委託別人來作呢,這實際上是MVC設計模式中的模塊分工問題,例如View對象它只負責顯示界面,而不須要進行數據的管理,數據的管理和邏輯是Controller的責任,因此此時View就應該將這個功能委託給Controller去實現,固然你做爲碼農強行讓View處理數據邏輯的任務,也不是不行,只是這就違背了MVC設計模式,項目小還好,隨着功能的擴展,咱們就會發現越寫越難寫;還有一種狀況,就是這件事情作不到,只能委託給其餘對象來作了,下面的例子中我會說明這種狀況。python
下面的代碼我想實現一個簡單的功能,場景描述以下:TableView上面有多個CustomTableViewCell,cell上面顯示的是文字信息和一個詳情Button,點擊button之後push到一個新的頁面。爲何說這個場景用到了代理delegate?由於button是在自定義的CustomTableViewCell上面,而cell沒有能力實現push的功能,由於push到新頁面的代碼是這樣的,ios
[self.navigationController pushViewController...];git
因此這時候CustomTableViewCell就要委託它所在的Controller去作這件事情了。github
按照個人編碼習慣,我喜歡把委託的協議寫在提出委託申請的類的頭文件裏面,如今的場景中是CustomTableViewCell提出了委託申請,下面是簡單的代碼,設計模式
@protocol CustomCellDelegate <NSObject>服務器
- (void)pushToNewPage;ide
@end 函數
@interface CustomTableViewCell : UITableViewCell學習
@property(nonatomic, assign) id<CustomCellDelegate> delegate;編碼
@property (nonatomic, strong) UILabel *text1Label;
@property(nonatomic,strong) UIButton *detailBtn;
上面的代碼在CustomTableViewCell.h中定義了一個協議CustomCellDelegate,它有一個須要實現的pushToNewPage方法,而後還要寫一個屬性修飾符爲assign、名爲delegate的property,之因此使用assign是由於這涉及到內存管理的東西,之後的博客中我會專門說明緣由。
接下來在CustomTableViewCell.m中編寫Button點擊代碼,
[self.detailBtn addTarget:self action:@selector(btnClicked:) forControlEvents:UIControlEventTouchUpInside];
對應的btnClicked方法以下,
- (void)btnClicked:(UIButton *)btn
{
if (self.delegate && [self.delegaterespondsToSelector:@selector(pushToNewPage)]) {
[self.delegate pushToNewPage];
}
}
上面代碼中的判斷條件最好是寫上,由於這是判斷self.delegate是否爲空,以及實現CustomCellDelegate協議的Controller是否也實現了其中的pushToNewPage方法。
接下來就是受到委託申請的類,這裏是對應CustomTableViewCell所在的ViewController,它首先要實現CustomCellDelegate協議,而後要實現其中的pushToNewPage方法,還有一點不能忘記的就是要設置CustomTableViewCell對象cell的delegate等於self,不少狀況下可能忘了寫cell.delegate = self;致使遇到問題不知雲裏霧裏。下面的關鍵代碼都是在ViewController.m中,
首先是服從CumtomCellDelegate協議,這個你們確定都知道,就像不少系統的協議,例如UIAlertViewDelegate、UITextFieldDelegate、UITableViewDelegate、UITableViewDatasource同樣。
@interface ViewController ()<CustomCellDelegate>
@property (nonatomic, strong) NSArray *textArray;
而後是實現CustomCellDelegate協議中的pushToNewPage方法,
- (void)pushToNewPage
{
DetailViewController*detailVC = [[DetailViewController alloc] init];
[self.navigationController pushViewController:detailVC animated:YES];
}
還有一個步驟最容易被忘記,就是設置CumtomTableViewCell對象cell的delegate,以下代碼,
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
static NSString *simpleIdentify = @"CustomCellIdentify";
CustomTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:simpleIdentify];
if (cell == nil) {
cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:simpleIdentify];
}
//下面代碼很關鍵
cell.delegate = self;
cell.text1Label.text = [self.textArray objectAtIndex:indexPath.row];
return cell;
}
經過cell.delegate = self;確保了CustomTableViewCell.m的判斷語句if(self.delegate && ...){}中得self.delegate不爲空,此時的self.delegate其實就是ViewController,cell對象委託了ViewController實現pushToNewPage方法。這個簡單的場景描述了使用代理的一種狀況,就是CustomTableViewCell沒有能力實現pushViewController的功能,因此委託ViewController來實現。
代碼在github能夠下載。
有什麼錯誤,還請你們指正。
----------------------------------------------下面是block的內容----------------------------------------------------
Block是一個C語言的特性,就像羣裏有人說的,它就是C語言的函數指針,在使用中最多的就是進行函數回調或者事件傳遞,好比發送數據到服務器,等待服務器反饋是成功仍是失敗,此時block就派上用場了,這個功能的實現也可用使用代理,這麼說的話,感受block是否是有點像代理了呢?
我以前接觸block,都是使用它做爲函數參數,當時感受不是很理解。如今在項目中,不少時候block做爲property,這樣更加簡單直接,想一想,其實property不就是定義的合成存儲的變量嘛,而block做爲函數參數也是定義的變量,因此做爲函數參數或者做爲property本質沒有區別。
看一看別人總結的block的語法吧,http://fuckingblocksyntax.com,這個連接亮了,fucking block syntax,操蛋的block語法啊。block有以下幾種使用狀況,
一、做爲一個本地變量(local variable)
returnType (^blockName)(parameterTypes) = ^returnType(parameters) {...};
二、做爲@property
@property (nonatomic, copy) returnType (^blockName)(parameterTypes);
三、做爲方法的參數(method parameter)
- (void)someMethodThatTakesABlock:(returnType (^)(parameterTypes))blockName;
四、做爲方法參數的時候被調用
[someObject someMethodThatTakesABlock: ^returnType (parameters) {...}];
五、使用typedef來定義block,能夠事半功倍
typedef returnType (^TypeName)(parameterTypes);
TypeName blockName = ^returnType(parameters) {...};
上面我也只是複製粘貼了一下,接下來仍是實現點擊CustomTableViewCell上面的Button實現頁面跳轉的功能,我以前不止一次的類比block就像delegate,這邊我也是思惟慣性,下面的內容我就當block爲代理,一些用詞描述仍是跟delegate差很少。首先,在提出委託申請的CustomTableViewCell中定義block的property,
@interface CustomTableViewCell : UITableViewCell
@property (nonatomic, strong) UILabel *text1Label;
@property (nonatomic, strong) UIButton *detailBtn;
//下面的定義,請看官們對比一下
/*delegate的定義 我沒有刪除,由於你們能夠類比了看下*/
@property (nonatomic, assign) id<CustomCellDelegate> delegate;
/*這裏定義了ButtonBlock*/
@property (nonatomic, copy) void (^ButtonBlock)();
@end
這裏用copy屬性來修飾ButtonBlock property,這個緣由,我會在之後的博客中做專門的解釋。
接下來在CustomTableViewCell中給它上面的detailBtn綁定點擊方法,
[self.detailBtn addTarget:self action:@selector(btnClicked:) forControlEvents:UIControlEventTouchUpInside];
而後是btnClicked方法的細節,我把delegate的內容也沒有刪除,就是給各位比較一下block和delegate的功能和語法的類似性,
- (void)btnClicked:(UIButton *)btn
{
//這是以前的delegate
if (self.delegate && [self.delegate respondsToSelector:@selector(pushToNewPage)]) {
[self.delegate pushToNewPage];
}
//這是如今咱們要說的block
if (ButtonBlock) {
ButtonBlock();
}
}
下面是一個關鍵性的地方,在ViewController2中設置其CustomTableViewCell的cell對象的ButtonBlock,也就是給它賦值,此處我仍是保留了cell.delegate = self;代碼,
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
NSString *blockIdentify = @"BlockIdentify";
CustomTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:blockIdentify];
if (cell == nil) {
cell = [[CustomTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:blockIdentify];
}
cell.text1Label.text = [self.textArray objectAtIndex:indexPath.row];
//delegate的不可缺乏的代碼,這裏放在這兒只是爲了給各位類比一下
cell.delegate = self;
//ButtonBlock不可缺乏的代碼
cell.ButtonBlock = ^{
[self pushToNewPage2];
};
return cell;
}
之因此cell.ButtonBlock = ^{};賦值,是由於咱們咱們是這樣定義ButtonBlock的,void (^ButtonBLock)(),表示無返回值無參數。
而後編寫pushToNewPage2方法,
- (void)pushToNewPage2
{
DetailViewController *detailVC = [[DetailViewController alloc] init];
[self.navigationController pushViewController:detailVC animated:YES];
}
大家看這個方法是否是與CustomCellDelegate協議中的pushToNewPage方法相似。而後在回過頭來類比同樣,是否是block就是精簡版的delegate,由於delegate設計模式要寫協議CustomCellDelegate,還有容易遺漏cell.delegate = self;可是block使用的時候就簡單多了。
最新的代碼我已經更新到github,很是簡單,有疑問或者有質疑的朋友能夠下載看看。
另:本人也是ios開發菜鳥,對於不少的細節瞭解不夠深刻,上面的代碼使用arc編碼,若是我寫的有什麼問題或者有什麼缺陷的話,還請大神給我指正。
很但願有各位開發者相互交流,提升技術,個人郵箱:zheniyerenrou@163.com。還有一個qq交流羣188647173,有興趣的能夠加進來討論討論、學習學習,也但願牛逼的大神到羣裏給咱們菜鳥指點指點。