文档章节

Swift讲解专题八——闭包

珲少
 珲少
发布于 2016/05/14 22:09
字数 1747
阅读 825
收藏 17
点赞 2
评论 3

Swift讲解专题八——闭包

一、引言

        Swift中的闭包是有一定功能的代码块,这十分类似于Objective-C中的block语法。Swift中的闭包语法风格十分简洁,其作用和函数的作用相似。

二、从一个系统函数看闭包

        Swift标准函数库中提供了一个sort排序函数,对于已经元素类型的数组,调用sort函数会进行重新排序并返回新的排序后的数组。这个sort函数可以接收一个返回值为Bool类型的闭包,来确定第一个元素是否排在第二个元素前面。代码示例如下:

var array = [3,21,5,2,64]
func func1(param1:Int,param2:Int) -> Bool {
    return param1>param2
}
//通过传入函数的方式
//array = [64,21,5,3,2]
array = array.sort(func1)
//通过闭包的方式
//array = [2,3,5,21,64]
array = array.sort({(param:Int,param2:Int)->Bool in
                        return param<param2
                    })

Swift语言有一个很显著的特点就是简洁,可以通过上下文推断出类型的情况一般开发都可以将类型的书写省略,这也是Swift语言设计的一个思路,由于闭包是作为函数的参数传入函数中的,因为函数参数的类型是确定,因此闭包的类型是可以被编译器推断出来的,开发者也可以将闭包的参数类型和返回值省略,上面的代码可以简写如下:

//将闭包的参数类型和返回值都省略
array = array.sort({(p1,p2) in return p1>p2})

实际上,如果闭包中的函数体只有一行代码,可以将return关键字也省略,这时会隐式的返回此行代码的值,如下:

array = array.sort({(p1,p2) in   p1>p2})

看到上面的表达式,是不是有点小震惊,闭包表达式竟然可以简写成这样!然而,你还是小看的Swift开发团队,后面的语法规则会让你明白什么是简洁的极致。可以看到上面的代码实现还是有3部分:参数和返回值,闭包关键字,函数体。参数和返回值即是参数列表,p1,p2,虽然省略了参数类型和返回值类型,但这部分的模块还在,闭包关键字即是in,它用来表示下面将是闭包的函数体,p1>p2即是函数体,只是这里省略了return关键字。闭包中既然参数类型和返回值类型编译器都可以自己推断出来,那么参数的数量编辑器也是可以自行推断的,因此,参数列表实际上也是多余的,闭包中会自动生成一些参数名称,和实际的参数数量向对应,例如上面sort函数中的闭包有两个参数,系统会自动生成$0和$1这两个参数名,开发者可以直接使用,因为参数列表都会省略了,那么也不再需要闭包关键字in来分隔参数列表与函数体,这时,闭包的写法实际上变成了如下的模样:

array = array.sort({$0<$1})

你没有看错,加上左右的大括号,一共7个字符,完成了一个排序算法。除了Swift,我不知道是否还有第二种语言可以做到。抛开闭包不说,Swift中还有一种语法,其可以定义类型的运算符方法,例如String类型可以通过=,<,>来进行比较,实际上是String类中实现了这些运算符方法,在某种意义上说,一个运算符即类似与一个函数,那么好了,sort函数中需要传入的方法对于某些类型来说实际上只是需要一个运算符,示例如下:

array = array.sort(>)

这次你可以真的震惊了,完成排序新算法只需要一个字符,不折不扣的一个字符。

三、Swift中闭包的更多特点

        Swift中的闭包还有一个有趣的特点,首先闭包是作为参数传入另一个函数中的,因此常规的写法是将闭包的大括号写在函数的参数列表小括号中,如果闭包中的代码很多,这时在代码结构上来看会变得并不太清晰,为了解决这个问题,Swift中这样规定:如果这个闭包参数是函数的最后一个参数,开发者可以将其拉出小括号,在函数尾部实现闭包代码,示例如下:

//闭包结尾
func func2(param1:Int,param2:()->Void)->Void{
    param2()
    print("调用了func2函数")
}
func2(0){
        print("闭包中的内容")
}

如果一个函数中只有一个参数,且这个参数是一个闭包,那么开发者使用闭包结尾这种写法,完全可以将函数的参数列表小括号也省略掉,示例如下:

func func3(param:()->Void)->Void{
    param()
    print("调用了func3函数")
}
func3{
    print("闭包中的内容")
}

Swift中还有一个闭包逃逸的概念,这个很好理解,当闭包作为参数传递进函数时,如果这个闭包只在函数中被使用,则开发者可以将这个闭包声明成非逃逸的,即告诉系统当此函数结束后,这个闭包的声明周期也将结束,这样做的好处是可以提高代码性能,将闭包声明称非逃逸的类型使用@noescape关键字,示例如下:

func func3(@noescape param:()->Void)->Void{
    param()
    print("调用了func3函数")
}
func3{
    print("闭包中的内容")
}

逃逸的闭包常用于异步的操作,例如这个闭包是异步处理一个网络请求,只有当请求结束后,闭包的声明周期才结束。非逃逸的闭包还有一个有趣的特点,在其内部如果需要使用self这个关键字,self可以被省略。

        闭包也可以被自动的生成,这种闭包被称为自动闭包,自动闭包可以自动将表达式封装成闭包,开发者不需要再写闭包的大括号格式,自动闭包不接收参数,返回值为其中表达式的值。示例如下:

//自动闭包演示
var list = [1,2,3,4,5,6]
//创建一个显式闭包
let closures = {
    list.removeFirst()
    list.append(7)
}
//将打印[1,2,3,4,5,6]
print(list)
//执行闭包
closures()
//将打印[2,3,4,5,6,7]
print(list)
func func4(closure:()->Void) -> Void {
    //执行显式的闭包
    closures()
}
func func5(@autoclosure auto:()->Void) -> Void {
    //执行自动闭包
    auto()
}
//显式闭包 需要大括号
func4(closures)
//将打印[3,4,5,6,7,7]
print(list)
//将表达式自动生成闭包
func5(list.append(8))
//将打印[3,4,5,6,7,7,8]
print(list)

自动闭包默认是非逃逸的,如果要使用逃逸的闭包,需要手动声明,如下:

func func5(@autoclosure(escaping) auto:()->Void) -> Void {
    //执行自动闭包
    auto()
}

专注技术,热爱生活,交流技术,也做朋友。

——珲少 QQ群:203317592

© 著作权归作者所有

共有 人打赏支持
珲少

珲少

粉丝 833
博文 372
码字总数 435295
作品 0
上海
iOS工程师
加载中

评论(3)

珲少
珲少

引用来自“珲少”的评论

引用来自“ios122”的评论

前端童鞋笑了: 闭包是一种技术,你说的闭包,其实就是匿名函数;block,最多算是 匿名函数的一个别名

各种需要的技术术语不同吧,官方的说法叫Closures,我相信大多数开发者都会翻译成闭包,至于block,是C和OC的语法,不会有任何意义,像C++中的lambda,当然你可以理解他们都是匿名函数,但是术语如此,本该如此

语言的技术术语,不会有任何异意,不好意思,输入法太不给力
珲少
珲少

引用来自“ios122”的评论

前端童鞋笑了: 闭包是一种技术,你说的闭包,其实就是匿名函数;block,最多算是 匿名函数的一个别名

各种需要的技术术语不同吧,官方的说法叫Closures,我相信大多数开发者都会翻译成闭包,至于block,是C和OC的语法,不会有任何意义,像C++中的lambda,当然你可以理解他们都是匿名函数,但是术语如此,本该如此
ios122
ios122
前端童鞋笑了: 闭包是一种技术,你说的闭包,其实就是匿名函数;block,最多算是 匿名函数的一个别名
Swift专题讲解十九——类型转换

Swift专题讲解十九——类型转换 一、类型检查与转换 在Objective-C和Java中,任何类型实例都可以通过强转使编译器认为它是另一种类型的实例,这么做其实是将所有的安全检查工作都交给了开发者...

珲少 ⋅ 2016/05/27 ⋅ 0

Swift专题讲解十六——ARC在Swift中的应用

Swift专题讲解十六——ARC在Swift中的应用 一、引言 ARC(自动引用计数)是Objective-C和Swift中用于解决内存管理问题的方案。在学习Objective-C编程时经常会学习到一个关于ARC的例子:在一个...

珲少 ⋅ 2016/05/20 ⋅ 1

Swift专题讲解二十——扩展

Swift专题讲解二十——扩展 一、简介 Swift中的扩展与Objective-C中的类别功能相似,扩展可以为一个已有的类、结构体、枚举或者协议添加新的属性或方法,与Objective-C的类别不同的是,Swift...

珲少 ⋅ 2016/05/29 ⋅ 0

Swift解读专题一——Swift2.2语言预览

专题一——Swift2.2语言预览 一、引言 本系列专题是我通过阅读Swift2.2语言开发文档,翻译总结加上自己的理解整理而成。其中大部分结构和内容都来自开发文档,有疏漏和错误之处,还望更多朋友...

珲少 ⋅ 2016/05/05 ⋅ 0

Swift专题讲解十五——类型构造

Swift专题讲解十五——类型构造 一、引言 构造是类、结构体、枚举在实例化中必须执行的过程,在构造过程中,类、结构体必须完成其中存储属性的构造。Swift中的构造通过构造方法来完成,和Obj...

珲少 ⋅ 2016/05/19 ⋅ 0

Swift专题讲解二十三——高级运算符

Swift专题讲解二十三——高级运算符 一、引言 除了前边博客中介绍的基本运算符外,Swift中还支持更多高级运算符,也支持开发者进行运算符的自定义。Swift中的算符运算符有一个特点,其不会产...

珲少 ⋅ 2016/05/31 ⋅ 1

Swift3.0带来的变化汇总系列三——函数和闭包写法上的微调

Swift3.0带来的变化汇总系列三——函数写法上的微调 一、函数方面 Swift3.0相比Swift2.2的版本在API上做了大量的修改,代码风格也更加统一。在函数方面,Swift3.0中做的最大修改是修改了内部...

珲少 ⋅ 2016/06/26 ⋅ 2

Swift讲解专题十三——下标访问

Swift讲解专题十三——下标访问 一、引言 在以前的博客中,讨论过在Objective-C中,通过下标的方式访问自定义数据模型的方法。Objective-C中主要是通过实现一系列方法来使自定义的数据类型支...

珲少 ⋅ 2016/05/17 ⋅ 1

Swift专题讲解十四——继承

Swift专题讲解十四——继承 一、引言 Swift中,一个类可以从另一个类继承方法、属性、下标及其他特性。当一个类继承于另一个类时,这个类被称为子类,所继承的类被称为父类。在Swift中,继承...

珲少 ⋅ 2016/05/18 ⋅ 1

Swift讲解专题五——集合类型

Swift讲解专题五——集合类型 一、引言 Swift中提供了3种集合类型,Array数据类型,Set集合类型,Dictionary字典类型。Array用于存放一组有序的数据,数据角标从0开始一次递增;Set用于存放一...

珲少 ⋅ 2016/05/11 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

ARMS: 原来实时计算可以这么简单!

摘要: 业务实时监控服务( ARMS)是一款阿里云应用性能管理(APM)类监控产品。借助本产品,您可以基于前端、应用、业务自定义等服务,迅速便捷地为企业构建秒级响应的业务监控能力。 业务实...

阿里云云栖社区 ⋅ 2分钟前 ⋅ 0

Monkey入门_琉璃

先下载android sdk安装配置好路径,然后adb shell 如果给你显示这个,说明目前没有有效的移动设备链接,可以开个安卓模拟器或者使用真机,usb或wifi链接到电脑都可以,打开usb调试模式;然后...

EvanDev ⋅ 3分钟前 ⋅ 0

Idea类注释模板

一、设置类注释模板 1.选择File–>Settings–>Editor–>File and Code Templates–>Includes–>File Header. 2.设置完成后,创建类时自动生成注释,效果如下。...

Clarence_D ⋅ 5分钟前 ⋅ 0

vuejs题

1、active-class是哪个组件的属性?嵌套路由怎么定义? 答:vue-router模块的router-link组件。 2、怎么定义vue-router的动态路由?怎么获取传过来的动态参数? 答:在router目录下的index.j...

自由小鸟 ⋅ 5分钟前 ⋅ 0

2018年社交系统ThinkSNS年中大促

致各大商企事业单位及粉丝用户: 为感谢大家对ThinkSNS品牌的关注与支持,2018年6月18日官方诚推出:年中大促,限时抢购活动! “ThinkSNS 年中大促,¥6.18超值特惠 名额有限,预购从速! ...

ThinkSNS账号 ⋅ 11分钟前 ⋅ 0

MYSQL主从复制搭建及切换操作(GTID与传统)

如下: MYSQL主从复制方式有默认的复制方式异步复制,5.5版本之后半同步复制,5.6版本之后新增GTID复制,包括5.7版本的多源复制。 MYSQL版本:5.7.20 操作系统版本:linux 6.7 64bit 1、异步...

rootliu ⋅ 11分钟前 ⋅ 0

Java强软弱虚引用Reference

Java强软弱虚引用Reference 本文目的:深入理解Reference 本文定位:学习笔记 学习过程记录,加深理解,提升文字组合表达能力。也希望能给学习Reference的同学一些灵感 源码说明 源码基于jdk...

lichuangnk ⋅ 14分钟前 ⋅ 0

plsql 表中字段及注释时为乱码

在windows中创 建一个名为“NLS_LANG”的系统环境变量,设置其值为“SIMPLIFIED CHINESE_CHINA.ZHS16GBK”, 然后重新启动 pl/sql developer,这样检索出来的中文内容就不会是乱码了。如...

江戸川 ⋅ 16分钟前 ⋅ 0

Docker创建JIRA 7.2.7中文破解版

1、介绍 1.1、什么是JIRA?   关于JIRA网上的介绍有很多,以下摘自百度百科:   JIRA是Atlassian公司出品的项目与事务跟踪工具,被广泛应用于缺陷跟踪、客户服务、需求收集、流程审批、任...

谢思华 ⋅ 20分钟前 ⋅ 0

Java Class 类使用

Java Class 类使用 我们可以通过已知的包名来获取到 Class 对象,从而可以通过反射动态的来操作对象。 获取Class有三种方式 //通过对象.class直接获取Class integerClass = Integer.class;...

gaob2001 ⋅ 25分钟前 ⋅ 0

没有更多内容

加载失败,请刷新页面

加载更多

下一页

返回顶部
顶部