![iOS开发:从零基础到精通](https://wfqqreader-1252317822.image.myqcloud.com/cover/796/26793796/b_26793796.jpg)
第5章 Objective-C语言特性
5.1 代码块
5.1.1 Block简介
代码块(Block)是从iOS 4开始引入的一个新特性。Block是对C语言的一个扩展,在Objective-C中完全支持。Block在现在的iOS开发中使用越来越普遍,因为Block使用起来非常强大,简单来说,Block就是封装了一组代码语句的对象,可以在任何时间执行。
1.Block简介
Block在官方文档中的定义是这样的:Block块是封装工作单元的对象,是可以在任何时间执行的代码段。其本质上是可移植的匿名函数,可以作为方法和函数的参数传入,可以从方法和函数中返回。
Block是对C语言的一种扩展,它并未作为标准的ANSI C所定义的部分,而是由苹果公司添加到语言中的。Block看起来更像是函数,可以给Block传递参数,Block也可以具有返回值。
在iOS 4以后,越来越多系统框架的API在使用Block。苹果对于Block的使用主要集中在以下几个方面:
- 完成处理(Completion Handlers);
- 通知处理(Notification Handlers);
- 错误处理(Error Handlers);
- 枚举(Enumeration);
- 动画与形变(View Animation and Transitions);
- 分类(Sorting);
- 线程管理(GCD/NSOperation)。
2.Block的定义与调用
块是以插入字符^开头,后面的一个括号内表示块所需要的参数,最后面的大括号中是块主体,最后以分号结束,如下代码所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T127_15227.jpg?sign=1739262437-kcxEJ2GfZ9nXNkj8yStfMr3VJFeRHmcw-0-ec8235ed823a6319f913c0f0f5d0f9bc)
同时,也可以将这个块赋值给一个变量printBlock,声明方式如下。其中,变量printBlock就是指向代码块的指针。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T128_15363.jpg?sign=1739262437-7brQpqCbLGw8BfS113hqaE1Nz343Y9Mf-0-b3c50da6c3ae82ef03b1c6ef8704c0c4)
如下代码,定义了一个变量printBlock,这个变量指向一个Block,Block位于等号右边。这个Block执行时,需要提供一个int型的参数,同时会返回一个int型的返回值。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T128_15365.jpg?sign=1739262437-O5F0S1pPEkppat3zVgeHMSASzsteBCnw-0-d43c93bab49fda687b0df7178299c5ad)
当需要调用已经定义的Block时,可以使用如下方式,和函数调用十分类似。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T128_15367.jpg?sign=1739262437-7ONeJq20kX5AeEgqE9OnzVWeLlvSTKd4-0-5271d7f87c0d2979bf068622791719e6)
3.把Block声明为类的属性
由于Block就是一个存储了一段代码的对象,因此,也可以把Block设置为某个类的属性。Block属性与其他类型的属性,如NSString、NSArray,没有什么本质区别,都可以使用点语法来对属性进行取值和赋值。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T128_15369.jpg?sign=1739262437-QBaeWdsYp4MUgAz34bZC3nr5pZLm46Yd-0-64666a90d2736d594cf8c0ee38290382)
注意:当声明一个Block类型的属性时,需要使用属性关键字copy。
在下面的示例代码中,添加了两个Block属性,在程序运行过程中,为两个Block属性进行赋值,即指定了一段代码,然后调用执行Block中的代码。
- 新建一个Single View Application工程,在ViewController.h文件中,声明两个Block属性。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T128_15371.jpg?sign=1739262437-6By7Nw7S7KybhfXrfTmI7gfaBtEG3vXW-0-0c99a88f2425e582758c99775fc62846)
- 在ViewController.m文件中,通过点语法为两个Block属性赋值,然后再调用Block中的代码。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T128_15373.jpg?sign=1739262437-u4pvqsHjBMcLjtYrhPE5cwXhERN3SiZN-0-d056cceb27ae8be65d686da1fe55ac48)
运行结果如图5-1所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P129_15493.jpg?sign=1739262437-5NdYcXbRw7ckBkPaIqafkal98ga7VN81-0-fbb62431812943174667f3c608997adf)
图5-1 运行结果
5.1.2 Block的参数与返回值
定义Block时,可以对Block的输入参数以及返回值的类型进行定义。可以有输入参数,也可以没有输入参数;可以设置一个输入参数,也可以设置多个参数;可以有返回值,也可以没有返回值。
1.无输入参数+无返回值
这种形式的Block,无须任何输入参数,并且无返回值,一般都是在该Block中完成一些动作。例如在UIView类中定义的animateWithDuration:animations:方法,当调用该方法时,在指定的时间(duration参数)内,完成Block((void (^)(void))animations参数)中定义的动画播放。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T129_15498.jpg?sign=1739262437-ia4f7JhxOOFdUugorXPi8VvXD30insHK-0-45408297903bcfbc7e13f2ff85417315)
这里也可以自定义一个无输入参数、无返回值的Block,如下所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T129_15500.jpg?sign=1739262437-HB17TVvtP4U8rIUv5MUnPwbAOKORSvYZ-0-66babda03aa1a5e0350d946e5ee5becf)
运行结果如图5-2所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P130_15602.jpg?sign=1739262437-RfMdfscdBKZf3j5plCtqVjJW3Sb5fAnz-0-caa031622a409e711ad4afb61c93c16e)
图5-2 运行结果
2.有输入参数+无返回值
这种形式的Block,有输入参数,但无返回值。一般都是在该Block中根据输入参数完成一些动作,例如,在AFNetworking框架提供的如下方法,需要传入3个Block参数。当获取到网络反馈的数据后,会调用一个Block,该Block没有返回值,但是存在两个参数,其中一个是从服务器获取的数据(responseObject)。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T130_15606.jpg?sign=1739262437-0NxK9CyVf4isryGmydSHLy63IGWmN46d-0-08dc8edcb174b27b0a1561b4c3fc0eea)
这里也可以自定义一个有输入参数、无返回值的Block,如下所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T130_15608.jpg?sign=1739262437-aYs2i5kL5Zh4vjToH5AKGWfPV51f6d3J-0-d470ca5f59a95a84b1f43e9b7ebc8ec7)
运行结果如图5-3所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P130_15610.jpg?sign=1739262437-fTvw22PdLyQ9HnavlN3GQRFyLZkgc205-0-c585a2b0d028e4b3054fb14bcd28c4e9)
图5-3 运行结果
3.有输入参数及返回值
Block中可以既有参数也有返回值,此时,需要在Block封装的代码中根据返回值的类型要求提供Block的返回值。例如,下方的示例代码中,blockWithOutputAndInput返回的是参数的平方(inputNum*inputNum)。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T131_82846.jpg?sign=1739262437-oPQSZlG3d5LCBAMjBUcRnA4cu9FrMx30-0-9f3f0be516df0c6567e14257bffdb871)
运行结果如图5-4所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P131_15753.jpg?sign=1739262437-6QDkHFNt4HrFxglq1iRAaMUkUSRgk9RM-0-148fb8c984296549290e461b91cad57d)
图5-4 运行结果
4.有多个输入参数
在Block中可以定义传入多个参数,多个参数之间使用逗号进行分隔。下面的示例代码中,Block需要传入两个Double类型的参数,这两个参数相乘后的乘积作为返回值。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T131_15757.jpg?sign=1739262437-1RztJZGECk8ARVRT6DWDLqKxHMk405nw-0-9a2a00d65e388b2411011c7883e3a600)
运行结果如图5-5所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P131_15759.jpg?sign=1739262437-KXsVO4tjLDbDcMBvsC2MbGxGE2CGAJDr-0-9476065c2c5a2f179c0f87e695f4fff4)
图5-5 运行结果
5.无输入参数+有返回值
最后一种情况是无参数有返回值的Block。如下所示,定义了一个Block,其没有参数,但是会有int型的返回值。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T132_15892.jpg?sign=1739262437-xy5SWvA5Vz40lOH6hDO3xwKZC0GnFwB2-0-84f7699724c6efa05000f6878e1f7c87)
运行结果如图5-6所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P132_82847.jpg?sign=1739262437-c5uK2HfIethVyuRPKAxBGkKowKBx5bP1-0-5b4ccfa4dc4f5233d75a85d704c8c06a)
图5-6 运行结果
5.1.3 操作Block外部的变量
在使用Block时,有时会涉及修改Block定义之外的对象,为了能够修改定义在Block之外的对象,必须在该对象声明时,添加_ _block关键字(两个下划线)。
1.访问Block之外的变量
如果在一个方法中声明了Block,那么Block中也可以访问在该方法中定义的变量,前提是该变量的定义在Block定义之前。如下所示,定义了一个int型的变量i,在名称为beginBlock的Block中,可以访问i值。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T132_15899.jpg?sign=1739262437-3Ei2AOZUVNnqtZNbzPBVX8svclAL0rFU-0-be1647a1868bc131031e24fca19b3cc9)
在上述代码中,Block可以访问i的值,但是当i值发生改变的时候(i=200),再次调用Block打印的还是原来的i值(100)。也就是说,在Block定义时,会“捕捉”一次Block中使用的对象i,当i发生变化的时候,不会影响已经“捕捉”到的值。
上述案例的运行结果如图5-7所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P133_15995.jpg?sign=1739262437-C3ELNE4eGxZ6gwAnAKKhKvnYPavK8yIb-0-79b72337a5452725986dd8e41fa33d75)
图5-7 运行结果
同时需要注意的是,此时在Block中是不能对i值进行修改的。假如修改Xcode会报错,如图5-8所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P133_82849.jpg?sign=1739262437-mNS1VFVxxioNH04Jcc1ZSd9VTUwSHMwq-0-560da2a64a5f292b17884206539fc658)
图5-8 程序报错提示
2.修改Block之外的变量
在Block中,假如需要更新在Block之外定义的变量,那么在定义变量时,必须加上_ _block关键字(两个下划线)。如果这样定义,以上面的代码为例,当i的值发生变化时,Block中“捕捉”的i值会随时变化。这个在实际开发中比较常用,如下所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T133_16003.jpg?sign=1739262437-a7ZBr9UN9EHpan4JkD6BCdCMmjlvBEcy-0-6cd12af6d163105fc1a27d7c04215a34)
运行结果如图5-9所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P134_16112.jpg?sign=1739262437-Q06dUh6mM7zPOecVnBS0q1iPb358AVL9-0-fa19866c9fbec75b7a076e7e3ed0fd61)
图5-9 运行结果
同时需要注意的是,此时在Block中可以对i的值进行修改,并且编译器也不会报错,如下所示。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T134_16116.jpg?sign=1739262437-VBAivoVDN8NY18bj41b833ysvVfoOti5-0-c6839e91a6f9a7fa2bc338875361ec22)
运行结果如图5-10所示。可以看到,每次执行Block后,i值都会在Block中修改为200,因此最后打印的i值是200。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P134_16118.jpg?sign=1739262437-xV68jvqE5lLQ5uTbhRYjZc0ZW47VOuvy-0-b97ee830b2877854a98854e0819b8a3a)
图5-10 运行结果
5.1.4 Block回调
在iOS的开发过程中,Block的回调使用非常普遍,也是Block的重要用法之一,在使用过程中经常可以用于替换代理的实现方法。例如,当一段动画播放完成后,执行一段代码,当得到请求的网络数据后,执行一段对数据的操作代码等。这些场景中,都使用到了Block的回调机制。Block的回调机制,可以使代码的编写变得十分清晰,提升了代码的可读性。
当需要定义回调Block时,通常情况下可以按照如下步骤进行:
- 定义带Block参数的方法。
- 设置Block的回调时机。
- 定义Block中需要执行的操作。
下面通过一个实际的例子来实践一下Block的回调实现方法。
- 创建一个Single View Application类型的工程。
- 定义带Block参数的方法。创建一个Task类,继承自NSObject。在Task.h文件中,添加如下的方法,在该方法中,设置一个Block作为参数。其中,(void(^)(void))表示为一个没有参数和返回值的Block。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T135_16229.jpg?sign=1739262437-5GUBhlDLn3AZmOpM5rKgM1zLYs4e7hye-0-a62cf9d8ac5938f03019c5aada375e90)
- 设置Block的回调时机。在Task.m文件中,实现该方法。下面的代码中,当方法被调用时,会打印一行Log,提示任务开始。3秒后,会调用Block中的代码。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T135_16231.jpg?sign=1739262437-ZA4nQHT9hTp67hSolQASKdZ3GLGR1Oq9-0-4137b1910b5dbdb130bb272a2cdb6046)
- 定义Block中需要执行的操作。在上面代码的实现过程中,最关键的是定义了Block的调用时机,但没有定义Block的代码内容。Block中的代码内容,可以在使用该方法时进行赋值。在工程的ViewController.m文件中,导入Task.h头文件,并添加下面的代码,当执行到Block时,打印一行日志,提示任务完成。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-T135_16233.jpg?sign=1739262437-CFEItoFqwpRzNqUsXNC9EBGkENKZoX5S-0-247d31acf1a4a36c324132f994d592ab)
运行结果如图5-11所示,通过两行日志执行的位置以及执行的时间,可以验证Block回调的使用方法。
![](https://epubservercos.yuewen.com/D4B438/15253388904120706/epubprivate/OEBPS/Images/Figure-P135_16235.jpg?sign=1739262437-TjhGm4o1QpVY9BjGkYUIt0KDyR42Zhm7-0-5326ca14857058a2eae723c5f929b235)
图5-11 运行结果