文档章节

iOS之轻松上手block

Align
 Align
发布于 2016/01/27 17:50
字数 1770
阅读 28
收藏 1

导语 不会使用block的iOS程序员,不是一个合格的程序员 学会了block,你再也不想用繁琐的代理 block没有你想象中的那么难,不要害怕,不要畏惧,勇敢尝试 笔者入行iOS时已经是ARC的天下,所以这里只说ARC环境下的使用 什么是block block其实就是一个代码块,把你想要执行的代码封装在这个代码块里,等到需要的时候再去调用。那block是OC对象吗?答案是肯定的

来自官方文档

笔者以英语3.9级的水平给大家翻译下,“block是一个OC对象,这意味着它能被添加到集合,比如NSArray、NSDictionary”

block的定义 block属性或变量 格式:返回值类型(^block名称)(参数列表) /定义属性,block属性可以用strong修饰,也可以用copy修饰 有小伙伴留言说苹果官方建议用copy,笔者查了下文档, 确实是这样的,不过笔者未测试出copy与strong的区别,大家喜欢啥就用啥吧/ @property (nonatomic, strong) void(^myBlock)();//无参无返回值 @property (nonatomic, strong) void(^myBlock1)(NSString *);//带参数 @property (nonatomic, strong) NSString *(^myBlock2)(NSString *);//带参数与返回值 //定义变量 void(^myBlock)() = nil;//无参无返回值 void(^myBlock1)(NSString *) = nil;//带参数 NSString *(^myBlock2)(NSString *) = nil;//带参数与返回值 block被当做方法的参数 格式:(block类型)参数名称

  • (void)test:(void(^)())testBlock//无惨无返回值
  • (void)test1:(void(^)(NSString *))testBlock//带参数
  • (void)test2:(NSString *(^)(NSString *))testBlock//带参数与返回值 使用typedef定义block typedef void(^myBlock)(); //以后就可以使用myBlock定义无参无返回值的block typedef void(^myBlock1)(NSString *); //使用myBlock1定义参数类型为NSString的block typedef NSString *(^myBlock2)(NSString *); //使用myBlock2定义参数类型为NSString,返回值也为NSString的block //定义属性 @property (nonatomic, strong) myBlock testBlock; //定义变量 myBlock testBlock = nil; //当做参数
  • (void)test:(myBlock)testBlock; block的赋值 格式:block = ^返回值类型(参数列表){}

没有参数没有返回值 myBlock testBlock = ^void(){ NSLog(@"test"); }; //没有返回值,void可以省略 myBlock testBlock1 = ^(){ NSLog(@"test1"); }; //没有参数,小括号也可以省略 myBlock testBlock2 = ^{ NSLog(@"test2"); }; 有参数没有返回值 myBlock1 testBlock = ^void(NSString *str) { NSLog(str); } //省略void myBlock1 testBlock = ^(NSString *str) { NSLog(str); } 有参数有返回值 myBlock2 testBlock = ^NSString *(NSString *str) { NSLog(str) return @"hi"; } //有返回值时也可以省略返回值类型 myBlock2 testBlock2 = ^(NSString *str) { NSLog(str) return @"hi"; } 实战 接下来,我们就结合一个实例程序,来看看block在实际开发中的简单使用

本案例涉及到两个控制器与一个Person类 联系人列表控制器:使用tableView展示联系人列表,称为A控制器 新建联系人控制器:创建新的联系人对象,称为B控制器 Person:联系人,有两个属性,name与phoneNumber 任务需求:点击A控制器右上角“新建”按钮跳到B控制器,B控制器添加联系人后,点击“保存”按钮返回A控制器,并将新添加的联系人展示到列表中

问题来了,如何将B控制器中的数据传递给A控制器呢? 那还不简单,A控制器直接把联系人数组传递给B控制器,B控制器新建联系人后添加到数组中,然后返回A控制器,在A控制器的viewWillAppear方法中刷新表格就OK了。

方法可行,但是不得不说,相当low,B控制器是用来添加联系人的,至于联系人数组什么情况,无需关心,所以,不要把数组传递给B控制器

B控制器要做的仅仅只是,新建联系人,然后把联系人对象传递给A控制器,至于A控制器拿到联系人后会做什么,那是A的事情,与B无关 看到这里,很多人可能已经想到了代理,没错,代理也可以实现,但...是...,B控制器定义协议,声明代理方法,A控制器设置代理,遵守协议,然后实现代理方法,B控制器在合适的地方调用代理方法,卧槽,好麻烦有木有,笔者都不想写代码了,还是回家种田去吧

好了不废话了,进入正题

使用block传递数据 在B控制器的.h文件中定义一个没有返回值,参数类型为Person的block属性 @property (nonatomic, strong) void(^saveBlock)(Person *); 在B控制器“保存”按钮的点击方法中调用block

  • (IBAction)save:(id)sender { //使用事先定义好的类方法创建Person对象 Person *person = [Person personWithName:_nameText.text phoneNumber:_phoneNumberText.text]; /*调用block之前最好先判断block是否为空,不为空才调用,否则程序崩溃/ //装逼写法 //!self.saveBlock? : self.saveBlock(person); //一般写法 if (self.saveBlock) { self.seveBlock(person); } [self.navigationController popViewControllerAnimated:YES]; } 在A控制器中,给B控制器的block属性进行赋值 //“新建”按钮点击执行的方法
    • (void)newContact { AddContactViewController *addVC = [[AddContactViewController alloc] init]; addVC.saveBlock = ^(Person *person){ //这里就可以拿到B控制器传递过来的person对象,添加到数组然后刷新表格 [self.contactList addObject:person]; [self.tableView reloadData]; }; [self.navigationController pushViewController:addVC animated:YES]; } 三步就搞定,很简单是不是,所以说,block并没有你们想想的那么复杂,自从笔者学会了block,就再也没用过代理,除了系统的。 block常见雷区---循环引用 使用block有一个特别要注意的地方,循环引用,何为循环引用?你引用我,我引用你,谁也不释放谁,对象无法销毁,占用内存

我们来看一个循环引用的一个例子

注意看控制台输出,当点击“取消”时,B控制器被销毁,dealloc方法被调用

把注释掉的代码打开,再运行

点击“取消”按钮,B被移除,但是dealloc方法没有调用,所以说,B控制器并没有销毁,why?

block对象赋值给了B控制器的属性,因此B会对block有一个强引用,而block中又用到了self(B控制器对象),block会对使用到的外部变量进行捕获,所以,block对B控制器也有一个强引用,最终造成循环引用,谁也无法释放

循环引用解决方法 循环引用如何解决?很简单,一行代码搞定

使用weakSelf(名称随便取的)替代self,block将不再对self进行强引用 图中__weak也可使用__unsafe_unretained,区别就是__weak修饰的指针,当对象销毁后,指针会被自动置为nil,而__unsafe_unretained修饰的指针,当对象销毁后会变成野指针,为了安全,推荐使用__weak

如此简单又好用的block,你是否已经学会了呢,如果觉得笔者的文章对你有帮助,请多多关注,你们的关注,将是笔者更新的动力,笔者将在下篇文章中,讲述block在内存中的情况,以及block对外界变量的捕获

© 著作权归作者所有

Align
粉丝 12
博文 65
码字总数 71695
作品 0
昌平
高级程序员
私信 提问
2018 iOS 面试题大全(补充完整版)

原文地址:2018 iOS 面试题大全 由于原作者并没有继续更新,这里我转过来继续更新下 这个栏目将持续更新--请iOS的小伙伴关注! 1、iOS 应用导航模式有哪些? 2、iOS 中持久化方式有哪些? 3、...

Theendisthebegi
2018/11/15
0
0
iOS | 用于解决循环引用的block timer

iOS 10的时候新增了一个带block的API: 苹果的官方文档里说,将这个timer本身作为参数传给block以此来避免循环引用: /// - parameter: block The execution body of the timer; the timer it...

无夜之星辰
2018/12/05
0
0
那些在学习iOS开发前就应该知道的事(part 1)

英文原文:Things I wish I had known before starting iOS development—Part 1 设计师设计出来了一个不错的引导界面,然而当我看到设计稿的时候,我们的app也没几天就要上线了。这个界面模...

TomatosX
2015/06/12
111
0
转:一套代码iOS、Android两端运行,Google Flutter意味着什么?

原文:https://www.toutiao.com/a6569388465538990600/?ttfrom=weixin&utmcampaign=clientshare×tamp=1529580533&app=newsarticle&utmsource=weixin&iid=35476649324&utmmedium=toutiao......

鸿蒙无上至尊
2018/06/22
0
0
iOS与JS交互之WKWebView-WKUIDelegate协议

级别:★★☆☆☆ 标签:「iOS与JS交互」「WKWebView与JS交互」「WKUIDelegate」 作者: Xs·H 审校: QiShare团队 先解释下标题:“iOS与JS交互”。iOS指原生代码(文章只有示例),JS指前端...

QiShare
2018/09/03
0
0

没有更多内容

加载失败,请刷新页面

加载更多

OSChina 周五乱弹 —— 葛优理论+1

Osc乱弹歌单(2019)请戳(这里) 【今日歌曲】 @这次装个文艺青年吧 :#今日歌曲推荐# 分享米津玄師的单曲《LOSER》: mv中的舞蹈诡异却又美丽,如此随性怕是难再跳出第二次…… 《LOSER》-...

小小编辑
48分钟前
46
2
nginx学习笔记

中间件位于客户机/ 服务器的操作系统之上,管理计算机资源和网络通讯。 是连接两个独立应用程序或独立系统的软件。 web请求通过中间件可以直接调用操作系统,也可以经过中间件把请求分发到多...

码农实战
今天
5
0
Spring Security 实战干货:玩转自定义登录

1. 前言 前面的关于 Spring Security 相关的文章只是一个预热。为了接下来更好的实战,如果你错过了请从 Spring Security 实战系列 开始。安全访问的第一步就是认证(Authentication),认证...

码农小胖哥
今天
14
0
JAVA 实现雪花算法生成唯一订单号工具类

import lombok.SneakyThrows;import lombok.extern.slf4j.Slf4j;import java.util.Calendar;/** * Default distributed primary key generator. * * <p> * Use snowflake......

huangkejie
昨天
14
0
PhotoShop 色调:RGB/CMYK 颜色模式

一·、 RGB : 三原色:红绿蓝 1.通道:通道中的红绿蓝通道分别对应的是红绿蓝三种原色(RGB)的显示范围 1.差值模式能模拟三种原色叠加之后的效果 2.添加-颜色曲线:调整图像RGB颜色----R色增强...

东方墨天
昨天
11
1

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部