注: 码字不易, 转载请标明出处, 欢迎讨论交流文章之外或深入技术
目录架构:
- 闭包简述
- 闭包表达式
- 闭包简写
- 尾随闭包
- 逃逸闭包
闭包简述
闭包跟 OC 中的 Block 的功能跟使用十分相似, 同样的, 特点: 可以作为独立的功能??槭褂酶?
闭包表达式
从声明来看, 其与 swift 函数声明非常相似, 参数与返回值的声明形式很相近;从用法来看, 其与 OC 的 Block 非常相似, 像属性一样使用与传递参数. 执行闭包的方式与 block/swift 方法一样, 传入参数直接调用即可.
参数以下代码为例:(对比 block 理解闭包更容易些)
等号左边为声明部分:
isAClusures : 闭包名称
(String) : 括号内为参数类型
String: 返回值类型
等号右边为实现部分:
(name:String) : 闭包参数
String: 返回值类型
in: 表示闭包的参数和返回值类型定义已经完成,闭包函数体即将开始, 引入函数体部分. in 前:闭包参数与返回值声明; in 后:函数体实现部分
/// 闭包声明 let isAClusures:(String) -> String = { (name:String)->String in ... return name } /// swift 的函数声明 func isAClusures(name:String) -> String { ... return name } /// block 声明 NSString *(^isABlock)(NSString *name) = ^NSString *(NSString *name) { ...; return name; };
闭包简写
// MARK:- 闭包的几种声明简写(对比声明的闭包是完整写法) // 1 >> 无参数无返回值, 可简化去掉参数与返回值以及关键字 in let isaClusures0 = { } // 2 >> 带参无返回值 let isaClusures1 = { (name:String) in } // 3 >> 带参有返回值 let isaClusures2 = { (name:String)->(String) in return name }
尾随闭包
概述:尾随闭包是一个书写在函数括号之后的闭包表达式,函数支持将其作为最后一个参数调用。
作用:使用尾随闭包时,不需要写出它的参数标签, 且把闭包作为函数最后一个参数传递时,使用尾随闭包可增强函数的可读性。一句话, 就是为了让代码书写更加精简,易读. 除此之外,没其他鸟用了,不理解也别纠结,收藏了慢慢理解,简书不死文章不删.
具体举两个例子说明,一个函数与闭包带参,一个函数与闭包都无参.
// 作为函数参数声明一 func followClusures0(name:String, age:Int, isaClusuresParams: (String, Int)->Void) -> Void { isaClusuresParams(name, 11) } // 作为函数参数声明二 (无函数参数, 无闭包参数) func followClusures2(isaClusuresParams:()->(String)) ->Void { }
对应的函数调用:→
// 1 >> 正常调用 followClusures0(name: "111", age: 11, isaClusuresParams: { (name2, age)->Void in print(name2) }) // 2 >> 尾随闭包 followClusures0(name: "111_1", age: 11) { (name2, age2) in print(name2) } // 3 >> 无参数 此时 直接省略() 省略 void in, 写法十分精简. 推荐使用 followClusures2 { return "value" }
逃逸闭包
概述:一个闭包作为参数传入一个函数,而闭包在函数返回之后才被调用,我们就说该闭包从函数中逃逸了. 声明一个含逃逸闭包参数的函数时,在闭包参数类型前使用@escaping关键字,即标明该闭包可逃逸.逃逸闭包需要显示的使用 self.
作用:在函数返回后, 在需要的时候调用. 较多用于异步操作或延时操作中. 相对于非逃逸闭包, 逃逸闭包更容易产生内存管理问题, 因为非逃逸闭包在函数返回时,会自动释放持有对象而不会产生内存问题. 逃逸闭包是在函数返回后调用, 容易造成内存泄漏. 个人理解, 当使用的闭包有内存问题需要注意时, 我们就应当使用逃逸闭包
代码如下:
var testString = "testString" func escapeClosures(isaEscapeClosures: @escaping (String)->(String)) { DispatchQueue.global().asyncAfter(deadline: (DispatchTime.now()+3)) { // 不显示的使用 self, 系统会报错 self.testString = "heiheihei" print("test:", self.testString) } self.isaEscapeClosures = isaEscapeClosures }
补充点:
- 在 OC 文件中导入 swift 文件, 直接导入build setting 自动生成的 .h文件.如下图:
下一次做一个混编传值的文章