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 的聲明 返回值類(lèi)型(^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 圖片上傳使用base64或者二進(jìn)制流上傳頭像功能
這篇文章主要介紹了iOS 圖片上傳使用base64或者二進(jìn)制流上傳頭像功能,需要的朋友可以參考下2017-09-09iOS中解決Xcode9的Log日志無(wú)法輸出中文的問(wèn)題小結(jié)
這篇文章主要介紹了iOS中解決Xcode9的Log日志無(wú)法輸出中文的問(wèn)題小結(jié),需要的朋友可以參考下2017-11-11iOS中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-05iOS開(kāi)發(fā)中使用Quartz2D繪圖及自定義UIImageView控件
這篇文章主要介紹了iOS開(kāi)發(fā)中使用Quartz2D繪圖及自定義UIImageView控件的方法,代碼基于傳統(tǒng)的Objective-C,需要的朋友可以參考下2015-11-11分享一個(gè)關(guān)于Storyboard 跳轉(zhuǎn)與傳值
近日不忙,給大家分享一個(gè)關(guān)于storyboard跳轉(zhuǎn)傳值的相關(guān)知識(shí),感興趣的朋友一起看看吧2015-12-12iOS統(tǒng)計(jì)代碼總行數(shù)的命令(便捷且簡(jiǎn)單)
在工作中有時(shí)候會(huì)需要統(tǒng)計(jì)我們工程的總代碼行數(shù),下面小編給大家分享一個(gè)超便捷超簡(jiǎn)單的方法,通過(guò)一行命令統(tǒng)計(jì)ios代碼總行數(shù),需要的朋友參考下吧2017-11-11iOS UILabel根據(jù)內(nèi)容自動(dòng)調(diào)整高度
這篇文章主要為大家詳細(xì)介紹了iOS UILabel根據(jù)內(nèi)容自動(dòng)調(diào)整高度,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06iOS左右滑動(dòng)標(biāo)簽頁(yè)導(dǎo)航的設(shè)計(jì)
這篇文章主要為大家詳細(xì)介紹了iOS左右滑動(dòng)標(biāo)簽頁(yè)導(dǎo)航的設(shè)計(jì)思路,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-06-06