本文簡介
本文主要是總結了block在開發中的幾種常用場景,如有不足的地方,還望指正。
Block簡介
Block是一種比較特殊的數據類型。它可以保存一段代碼,在合適的時候取出來調用。
block的常用場景
1 . 保存代碼
- 例如A界面中,幾個cell的樣式一樣,但點擊cell所觸發的行為不一樣,我們可以在cell的模型中聲明一個block類型的屬性來記錄cell的點擊行為。
CellItem.h
#import <Foundation/Foundation.h>
typedef void(^BlockName)();
@interface CellItem : NSObject
@property (nonatomic ,strong) NSString *title;
/**
* 用這個block屬性來記錄cell的點擊行為
*/
@property (nonatomic, strong) void(^operationBlock)();
@end
ViewController.m
- (void)viewDidLoad {
[super viewDidLoad];
// 創建模型,模型保存了對應cell的處理事件
CellItem *item = [[CellItem alloc] init];
item.title = @"打電話";
item.operationBlock = ^{
NSLog(@"打電話");
};
CellItem *item1 = [[CellItem alloc] init];
item1.title = @"發短信";
item1.operationBlock = ^{
NSLog(@"發短信");
};
CellItem *item2 = [[CellItem alloc] init];
item2.title = @"發郵件";
item2.operationBlock = ^{
NSLog(@"發郵件");
};
//將模型放入數組中
[self.arr addObject:item];
[self.arr addObject:item1];
[self.arr addObject:item2];
}
// 選中一行cell就會調用
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
// 獲取對應的模型
// 獲取模型
CellItem *item = self.arr[indexPath.row];
// 調用模型中保存的代碼
if (item.operationBlock) {
item.operationBlock();
}
2 . 逆向傳值
逆向傳值可以用代理實現也可以用block實現,用block要比代理簡單的多,但過程實際上很相似。
例如點擊A控制器的view,modal出一個B控制器,然后點擊B控制器的view,向A控制器傳一個值。通常用代理的做法就是在B聲明代理,A實現代理方法,在B中調用這個代理方法,用block實現其實也一樣。
首先我們要在B控制器中聲明一個有參數的block類型的屬性
ModalViewController.h
// 傳值:需要傳值的時候,再去調用
// 這里就相當于聲明一個代理
@property (nonatomic ,strong) void(^valueBlcok)(NSString *str);
- 然后在A控制器里面定義這個block類型的屬性
AViewController.m
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
ModalViewController *modalVc = [[ModalViewController alloc] init];
// 定義這個block,就相當于實現代理
// block本質就是保存代碼,這里用block保存一段NSLog代碼,等調用這個block時,才會執行這段代碼
modalVc.valueBlcok = ^(NSString *str){
NSLog(@"ViewController拿到%@",str);
};
[self presentViewController:modalVc animated:YES completion:nil];
}
- 最后我們要在B控制器中調用這個屬性
- ModalViewController.m
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
// 傳值:調用block
// 和代理類似,判斷block是否存在,有的話,傳值,調用blocK,也就是執行了A定義block時保存的代碼,完成了逆傳
if (_valueBlcok) {
_valueBlcok(@"123");
}
}
3 . 作為參數使用
- 蘋果有很多方法都是以block作為參數的,我們調用這個方法,在block的代碼塊里寫入要要執行的代碼,但是什么時候調用我們寫入的代碼卻是由內部決定的。
- 例如我們需要一個可以自由傳入計算式子,然后輸出結果的方法
- 首先,創建一個類,提供一個帶有block參數的方法
CalculatorManager.h
#import <Foundation/Foundation.h>
@interface CalculatorManager : NSObject
/** 計算結果 */
@property(nonatomic ,assign) int result;
// 提供一個計算方法
// block參數 void(^)()
// 方法括號:放參數變量類型
- (void)calculator:(int(^)(int result))block;
- 在.m文件中具體實現這個方法
CalculatorManager.m
#import "CalculatorManager.h"
@implementation CalculatorManager
- (void)calculator:(int (^)(int))block
{
// 將外界傳入的block保存的代碼在這里執行
_result = block(_result);
}
- 在控制器中調用這個方法
ViewController.m
- (void)viewDidLoad {
[super viewDidLoad];
// 創建計算器管理者
CalculatorManager *mgr = [[CalculatorManager alloc] init];
// 計算
[mgr calculator:^(int result){
// 我們決定計算方式,什么時候調用由內部決定
// 計算結果值
result += 5;
return result;
}];
NSLog(@"%d",mgr.result);
}
- 輸出結果為5
4 . 作為返回值
-
這里用到了鏈式編程思想
- 鏈式編程思想:可讀性極高,把一些方法通過
.
連接起
來,例如Masonry
框架中就用到這種思想:make.top.equalTo(superview.mas_top).with.offset(padding.top);
- 鏈式編程思想:可讀性極高,把一些方法通過
這里舉一個簡單的例子來闡述其原理
繼續用上一個的計算器類,我們創建一個加方法
CalculatorManager.h
#import <Foundation/Foundation.h>
@interface CalculatorManager : NSObject
@property(nonatomic ,assign) int result;
// 返回block void(^)()
// 將本類作為block返回值類型
- (CalculatorManager *(^)(int a))add;
@end
- 在.m中具體實現
CalculatorManager.m
- (CalculatorManager * (^)(int a))add
{
// 返回一個block塊
return ^(int a){
// 在block塊里,首先實現加法計算
// 最后返回self
_result += a;
return self;
};
}
- 最后在控制器里面調用
ViewController.m
- (void)viewDidLoad {
[super viewDidLoad];
CalculatorManager *mgr = [[CalculatorManager alloc] init];
// 調用屬性的get方法
// [[[mgr add:5] add:5] add:5];
// 因為返回值是self,所以可以連續調用,同時通過block保存的代碼塊可以保存下result的值
// 可以用點語法進行連續調用
mgr.add(5).add(5).add(5);
NSLog(@"%d",mgr.result);
}
- 輸出結果是15