你好,歡迎來到IOS教程網

 Ios教程網 >> IOS編程開發 >> IOS開發綜合 >> iOS中Block的回調使用和解析詳解

iOS中Block的回調使用和解析詳解

編輯:IOS開發綜合

Block 回調實現

先跟著我實現最簡單的 Block 回調傳參的使用,如果你能舉一反三,基本上可以滿足了 OC 中的開發需求。已經實現的同學可以跳到下一節。

首先解釋一下我們例子要實現什麼功能(其實是爛大街又最形象的例子):

有兩個視圖控制器 A 和 B,現在點擊 A 上的按鈕跳轉到視圖 B ,並在 B 中的textfield 輸入字符串,點擊 B 中的跳轉按鈕跳轉回 A ,並將之前輸入的字符串

顯示在 A 中的 label 上。也就是說 A 視圖中需要回調 B 視圖中的數據。

想不明白的同學可以看一看最終實現的效果圖:

這裡不再對 Block 的語法做說明了。

首先,我們需要定義兩個試圖控制器 AViewController BViewController,現在我們需要思考一下,Block 應該在哪裡定義呢?

我們可以簡單地這樣思考,需要回調數據的是 A 視圖,那麼 Block 就應該在 B 中定義,用於獲取傳入回調數據。

因此我們在 BViewController.h 中定義如下:

//BViewController.h
#import <UIKit/UIKit.h>

typedef void(^CallBackBlcok) (NSString *text);//1

@interface BViewController : UIViewController

@property (nonatomic,copy)CallBackBlcok callBackBlock;//2
@end

在這裡,代碼 1 用 typedef 定義了 void(^) (NSString *text)的別名為 CallBackBlcok 。這樣我們就可以在代碼 2 中,使用這個別名定義一個 Block 類型的變量 callBackBlock

在定義了 callBackBlock 之後,我們可以在 B 中的點擊事件中添加 callBackBlock 的傳參操作:

//BViewController.m

- (IBAction)click:(id)sender {
 self.callBackBlock(_textField.text); //1
 [self.navigationController popToRootViewControllerAnimated:YES];
}

這樣我們就可以在想要獲取數據回調的地方,也就 A 的視圖中調用 block:

// AViewController.m
- (IBAction)push:(id)sender {
 BViewController *bVC = [self.storyboard instantiateViewControllerWithIdentifier:@"BViewController"];

 bVC.callBackBlock = ^(NSString *text){ // 1

  NSLog(@"text is %@",text);

  self.label.text = text;

 };
 [self.navigationController pushViewController:bVC animated:YES];
}

代碼 1 中,通過對回調將 B 中的數據傳遞到代碼塊中,並賦值給 A中的 label,實現了整個回調過程。

上例是通過將 block 直接賦值給 block 屬性,也可以通過方法參數的方式傳遞 block 塊。

關於 Block 的疑惑

到目前為止,一切看起來都很美好(如果你照著上面的例子做的話),功能正常, A 視圖中也獲取到數據了。但是某些人可能就要說了,你的代碼有問題,你的思路有問題,你這是誤人子弟。

是的,代碼的確還有問題,第一個問題就是循環引用的問題,在 A 視圖的block 代碼塊中:

bVC.callBackBlock = ^(NSString *text){
  NSLog(@"text is %@",text);  
  self.label.text = text;  
 };

代碼 self.label.text = text; ,在 Block 中引用 self ,也就是 A ,而 A 創建並引用了 B ,而 B 引用 callBackBlock,此時就形成了一個循環引用,而編譯器也不會報任何錯誤,我們需要非常小心這個問題(面試百分百問到我會亂說?)。此時我們通常的解決方法是使用弱引用來解除這個循環:

 __weak AViewController *weakSelf = self;
 bVC.callBackBlock = ^(NSString *text){ 
  NSLog(@"text is %@",text); 
//  self.label.text = text; 
  weakSelf.label.text = text;
 };

第二個問題是我自己對 Block 的理解不到位,我們都知道 Block 能截取自動變量,並且是不能在 Block 塊中進行修改的(除非用__block修飾符),但是很明顯 weakSelf.label.text的值被修改了,並且沒有用__block修飾符, 這是為什麼呢?因為 label 是個全局變量,而如果像如下的局部變量 a 是不能修改的,編譯器也會報錯:


局部變量

通過這個小例子發現的兩個問題,也算是值得了。

Block 為什麼能實現神奇的回調

在這裡我不會說什麼實現原理,僅僅是個人對 Block 能實現神奇回調的理解,有錯誤的地方請大家指出。

在先前使用 Block 的過程中,雖然會使用,但是總是有一個疑惑,簡單說來就是:

為什麼在 A 中的 block 塊能調用到 B 中的數據?

回顧一下我們在 B 中所實現的代碼,不外乎定義了一個 Block 變量,並在適當的時候傳入參數,那麼為什麼在調用了  self.callBackBlock(_textField.text) 之後,值就神奇傳到了 A 中的 Block 塊了呢?

通過整理使用的過程,我發現是我們的思維陷入了誤區(可能是我個人),我們認為在 B 中傳入 _textField.text 參數之後, A 中的 Block 塊就可以獲取到值。雖然思路是對的,但其實是不完整,導致我們形成了回調的數據是通過某種底層實現傳遞過去的錯覺,這就使得我們認為這不需要深究。

事實是,通過簡單的整理我們可以發現完整的回調流程應該是這樣的:


回調流程

block 代碼塊賦值給 bVC.callBackBlock,此時 callBackBlock 的指針就指向這個代碼塊。

調用 callBackBlock(NSString *text)

由於 callBackBlock 的指針是指向 A 中的 block 代碼塊,因此執行代碼塊的代碼,實現回調。

很顯然之前我忽略了代碼塊賦值給 callBackBlock 的這個操作(羞愧)。

現在再通過一段代碼可以更清晰地理解這個原理:

 bVC.callBackBlock = ^(NSString *text){ //1
  NSLog(@"text is %@",text);
 };
 bVC.callBackBlock = ^(NSString *text){ //2
  NSLog(@"text b is %@",text);
 };

上述代碼中,我們對 callBackBlock進行了兩次賦值,結果會怎麼樣呢?


two block

可以看出來,Block 的回調只對代碼 2 生效,因為callBackBlock的指針最後指向了代碼 2 的代碼塊。所以並沒有什麼神奇的魔法,也沒什麼隱藏的底層機制(這裡指的是方便理解的底層)讓你可以帶著疑惑去使用它。

總結

我這個人學習方法,總結起來就是看到新技術,先在自己的代碼裡跑一遍,能跑通,並且使用起來沒有什麼難度,就基本不會深究了。但是自我反思過,這樣的學習方法是很不對的,寫代碼不能不求甚解,如果想要有所突破,不想局限於碼農,一定要深入探究一下實現的機制,最起碼要保證不帶著疑惑去使用。以上就是這篇文章的全部內容,希望能對大家的學習或者工作帶來一定的幫助,如果有疑問大家可以留言交流。

  1. 上一頁:
  2. 下一頁:
蘋果刷機越獄教程| IOS教程問題解答| IOS技巧綜合| IOS7技巧| IOS8教程
Copyright © Ios教程網 All Rights Reserved