iOS(閉包)block傳值詳解
在iOSAPP開(kāi)發(fā)的過(guò)程中 我們會(huì)用到很多需要傳值的地方 傳值的方式也多種多樣 有:代理傳值、通知傳值、KVC、KVO、block、單例 等。其中block 因?yàn)槠浜?jiǎn)潔實(shí)用規(guī)范的代碼 無(wú)疑是大牛們傳值的不二選擇 但對(duì)于初學(xué)者來(lái)說(shuō)要理解并能運(yùn)用 起初確實(shí)有些難以理解 以下我將細(xì)細(xì)的介紹下block
首先我總結(jié)了一下block的公式:
- 步驟1.block 的聲明 返回值類型(^block 的名字)(參數(shù)列表);
- 步驟2.block 實(shí)現(xiàn) block的名字 = ^(參數(shù)列表)(){};
- 步驟3.block 的調(diào)用 block的名字();
下面是一個(gè)簡(jiǎn)單的block
// 聲明
void(^blockName)(int num,NSString *string);
// 實(shí)現(xiàn)
blockName = ^(int num,NSString *string)
{
NSLog(@"%d,%@",num,string);
};
// 調(diào)用
blockName(520,@"phyone_");
}
由該例可以看出block的代碼執(zhí)行順序和其他的代碼執(zhí)行順序不一樣 它是先執(zhí)行聲明代碼 然后執(zhí)行調(diào)用代碼 最后才執(zhí)行實(shí)現(xiàn)代碼 由此我們利用block的回調(diào)性質(zhì) 達(dá)到我們傳值效果 當(dāng)然這只是在一個(gè)頁(yè)面的一段代碼 只是說(shuō)明了它具有傳值的作用 下面我們來(lái)試試 多頁(yè)面之間的block反向傳值 由于是反向傳值 所以我們創(chuàng)建兩個(gè)ViewController (ViewController,Next_ViewController)先由ViewController跳到Next_ViewController里將里面的值反向傳給ViewController
以下是代碼
ViewController.m里面
- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor purpleColor];
UILabel *lable = [[UILabel alloc]initWithFrame:CGRectMake(100, 200, 200, 100)];
lable.backgroundColor = [UIColor brownColor];
[self.view addSubview:lable];
UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tap)];
[self.view addGestureRecognizer:tap];
}
- (void)tap
{
Next_ViewController *NVC = [[Next_ViewController alloc]init];
NVC.block = ^(NSString *content){
NSLog(@"%@",content);
UIButton *button = [[UIButton alloc]initWithFrame:CGRectMake(100, 200, 100, 100)];
[button setTitle:content forState:UIControlStateNormal];
button.backgroundColor = [UIColor orangeColor];
[self.view addSubview:button];
};
[self presentViewController:NVC animated:YES completion:nil];
}
Next_ViewController.h里面
//block 是分配在棧里面的用copy @property(nonatomic,copy) void(^block)(NSString *content);
Next_ViewController.m里面
<p style="margin-top: 0px; margin-bottom: 0px; font-family: Menlo;"><span style="font-size:12px;">{</span></p>
<p style="margin-top: 0px; margin-bottom: 0px; font-family: Menlo;"><span style="font-size:12px;"> void(^imageNameBlock_1)(NSString *imageName);</span></p>
<p style="margin-top: 0px; margin-bottom: 0px; font-family: Menlo;"><span style="font-size:12px;"> void(^lableBlock)(NSString *tontentText);</span></p>
<p style="margin-top: 0px; margin-bottom: 0px; font-family: Menlo;"><span style="font-size:12px;">}</span></p>
- (void)viewDidLoad {
[super viewDidLoad];
self.view.backgroundColor = [UIColor orangeColor];
UIButton *button = [UIButton buttonWithType:UIButtonTypeCustom];
button.frame = CGRectMake(100, 200, 100, 50);
[button setTitle:@"123456" forState:UIControlStateNormal];
button.backgroundColor = [UIColor redColor];
[button addTarget:self action:@selector(buttonAction:) forControlEvents:UIControlEventTouchUpInside];
[self.view addSubview:button];
}
- (void)buttonAction:(UIButton *)sender
{
// 點(diǎn)擊按鈕 返回上一個(gè)頁(yè)面 同時(shí)通過(guò)block 把這個(gè)值傳到上一個(gè)頁(yè)面
self.block(sender.titleLabel.text);
[self dismissViewControllerAnimated:nil completion:nil];
}
另外:block 還有兩個(gè)問(wèn)題
1.怎么改變block里面的值
用上面的代碼 在ViewController.m viewDidLoad 里面 添加代碼
1.局部變量
定義一個(gè)圖片名字的Block
*/
// 局部變量
void(^imageNameBlock)(NSString *imageName);
// 如果想修改Block里面的值 需添加__block修飾
// __block UIImage *image;
imageNameBlock = ^(NSString *imageName)
{
image = [UIImage imageNamed:imageName];
self.view.backgroundColor = [UIColor colorWithPatternImage:image];
};
imageNameBlock(@"17.jpg");
就會(huì)出現(xiàn) Use of undeclared identifier 'image' 錯(cuò)誤情況 原因是我們?cè)噲D改變block里面的值 解決辦法就是 添加 __block 修飾(解注釋
__block UIImage *image;) 就行了
2.block的循環(huán)引用
用上面的代碼 在ViewController.m viewDidLoad 里面 添加代碼
// 全局變量
// block循環(huán) 引用 解決 我們用弱引用 __block
__weak ViewController *VC = self;
__block UIImage *image = nil;
imageNameBlock_1 = ^(NSString *imageName)
{
image = [UIImage imageNamed:imageName];
VC.view.backgroundColor = [UIColor colorWithPatternImage:image];
};
UILabel *lable = [[UILabel alloc]initWithFrame:CGRectMake(100, 200, 200, 100)];
lable.backgroundColor = [UIColor brownColor];
[self.view addSubview:lable];
UITapGestureRecognizer *tap = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tap)];
[self.view addGestureRecognizer:tap];
再添加兩個(gè)方法
- (void)loadData
{
imageNameBlock_1(@"17.jpg");
}
- (void)viewWillAppear:(BOOL)animated
{
[self loadData];
}
當(dāng)出現(xiàn)循環(huán)引用是我們用__weak修飾下 但具體機(jī)制還不太懂 希望高人指點(diǎn),感謝大家對(duì)腳本之家的支持。
相關(guān)文章
詳解ios中的SQL數(shù)據(jù)庫(kù)文件加密 (使用sqlcipher)
本篇文章主要介紹了ios中的SQL數(shù)據(jù)庫(kù)文件加密 (使用sqlcipher),具有一定的參考價(jià)值,這里整理了詳細(xì)的代碼,感興趣的小伙伴們可以參考一下。2016-12-12
ios學(xué)習(xí)筆記之基礎(chǔ)數(shù)據(jù)類型的轉(zhuǎn)換
在編碼過(guò)程中,數(shù)據(jù)的處理是必要的。眾多數(shù)據(jù)中,NSString、NSData、NSArray、 NSDictionary等數(shù)據(jù)類型是常用的,對(duì)付它們?nèi)菀?,但是在多個(gè)數(shù)據(jù)類型之間轉(zhuǎn)換就需要技巧了。本文主要給大家介紹ios中基礎(chǔ)數(shù)據(jù)類型的轉(zhuǎn)換,有需要的下面來(lái)一起看看吧。2016-11-11
使用Reachability類判斷iOS設(shè)備的當(dāng)前網(wǎng)絡(luò)連接類型
這篇文章主要介紹了使用Reachability類判斷iOS設(shè)備的當(dāng)前網(wǎng)絡(luò)連接類型,這里開(kāi)發(fā)語(yǔ)言為傳統(tǒng)的Objectice-C,需要的朋友可以參考下2016-02-02
IOS Object-C 中Runtime詳解及實(shí)例代碼
這篇文章主要介紹了IOS Object-C 中Runtime詳解及實(shí)例代碼的相關(guān)資料,OC中的對(duì)象其實(shí)在Runtime中都會(huì)用結(jié)構(gòu)體來(lái)表示,這個(gè)結(jié)構(gòu)體中包含了類名、成員變量列表、方法列表、協(xié)議列表、緩存等,需要的朋友可以參考下2017-03-03
iOS中UIActivityIndicatorView的用法及齒輪等待動(dòng)畫(huà)實(shí)例
UIActivityIndicatorView活動(dòng)指示器最常見(jiàn)的用法便是用來(lái)制作那個(gè)程序中的齒輪轉(zhuǎn)動(dòng)的等待效果,接下來(lái)我們回來(lái)簡(jiǎn)單整理iOS中UIActivityIndicatorView的用法及齒輪等待動(dòng)畫(huà)實(shí)例:2016-05-05
iOS 自定義狀態(tài)欄和導(dǎo)航欄詳細(xì)介紹
這篇文章主要介紹了iOS 自定義狀態(tài)欄和導(dǎo)航欄詳細(xì)介紹的相關(guān)資料,需要的朋友可以參考下2016-11-11
深入分析iOS應(yīng)用中對(duì)于圖片緩存的管理和使用
這篇文章主要介紹了iOS應(yīng)用中對(duì)于圖片緩存的管理和使用,實(shí)例代碼為傳統(tǒng)的Objective-C,需要的朋友可以參考下2016-04-04

