简介:本文主要讲:函数的定义,外部参数的用处,无返回类型的三种函数定义方式html
闭包的定义,闭包的概念和用法,尾随闭包的写法,解除循环引用的方法swift
func 函数名(行参列表) -> 返回值 {代码实现}
let result = 函数名(值1, 参数2: 值2...)
func sum(a: Int, b: Int) -> Int { return a + b } let result = sum(10, b: 20)
func demo(str: String) -> Void { print(str) } func demo1(str: String) -> () { print(str) } func demo2(str: String) { print(str) } demo("hello") demo1("hello world") demo2("olleh")
func 函数名(外部参数名 形式参数名: 形式参数类型) -> 返回值类型 { // 代码实现 }
func sum1(num1 a: Int, num2 b: Int) -> Int { return a + b } sum1(num1: 10, num2: 20)
与 OC 中的 Block 相似,闭包
主要用于异步操做执行完成后的代码回调,网络访问结果以参数的形式传递给调用方网络
//: 定义一个 sum 函数 func sum(num1 num1: Int, num2: Int) -> Int { return num1 + num2 } sum(num1: 10, num2: 30) //: 在 Swift 中函数自己就能够看成参数被定义和传递 let mySum = sum let result = mySum(num1: 20, num2: 30)
in
用于区分函数定义和代码实现//: 闭包 = { (行参) -> 返回值 in // 代码实现 } let sumFunc = { (num1 x: Int, num2 y: Int) -> Int in return x + y } sumFunc(num1: 10, num2: 20)
参数/返回值/in
通通均可以省略
let demoFunc = { print("hello") }
func loadData() { dispatch_async(dispatch_get_global_queue(0, 0), { () -> Void in print("耗时操做 \(NSThread .currentThread())") }) }
}
的位置func loadData() { dispatch_async(dispatch_get_global_queue(0, 0)) { () -> Void in print("耗时操做 \(NSThread .currentThread())") } }
func loadData() { dispatch_async(dispatch_get_global_queue(0, 0)) { print("耗时操做 \(NSThread .currentThread())") } }
override func viewDidLoad() { super.viewDidLoad() loadData { print("完成回调") } } // MARK: - 自定义闭包参数 func loadData(finished: ()->()) { dispatch_async(dispatch_get_global_queue(0, 0)) { print("耗时操做 \(NSThread.currentThread())") dispatch_sync(dispatch_get_main_queue()) { print("主线程回调 \(NSThread.currentThread())") // 执行回调 finished() } } }
override func viewDidLoad() { super.viewDidLoad() loadData4 { (html) -> () in print(html) } } /// 加载数据 /// 完成回调 - 传入回调闭包,接收异步执行的结果 func loadData4(finished: (html: String) -> ()) { dispatch_async(dispatch_get_global_queue(0, 0)) { print("加载数据 \(NSThread.currentThread())") dispatch_sync(dispatch_get_main_queue()) { print("完成回调 \(NSThread.currentThread())") finished(html: "<h1>hello world</h1>") } } }
NetworkTools
对象class NetworkTools: NSObject { /// 加载数据 /// /// - parameter finished: 完成回调 func loadData(finished: () -> ()) { print("开始加载数据...") // ... finished() } deinit { print("网络工具 88") } }
NetworkTools
而且加载数据class ViewController: UIViewController { var tools: NetworkTools? override func viewDidLoad() { super.viewDidLoad() tools = NetworkTools() tools?.loadData() { print("come here \(self.view)") } } /// 与 OC 中的 dealloc 相似,注意此函数没有() deinit { print("控制器 88") } }
运行不会造成循环引用,由于 loadData 执行完毕后,就会释放对 self 的引用闭包
NetworkTools
,定义回调闭包属性/// 完成回调属性 var finishedCallBack: (()->())? /// 加载数据 /// /// - parameter finished: 完成回调 func loadData(finished: () -> ()) { self.finishedCallBack = finished print("开始加载数据...") // ... working() } func working() { finishedCallBack?() } deinit { print("网络工具 88") }
运行测试,会出现循环引用异步
/// 相似于 OC 的解除引用 func demo() { weak var weakSelf = self tools?.loadData() { print("\(weakSelf?.view)") } }
loadData { [weak self] in print("\(self?.view)") }
loadData { [unowned self] in print("\(self.view)") }
Swiftasync
[weak self]
self
是可选项,若是self已经被释放,则为nil
[unowned self]
self
不是可选项,若是self已经被释放,则出现野指针访问
Objcide
__weak typeof(self) weakSelf;
self
已经被释放,则为nil
__unsafe_unretained typeof(self) weakSelf;
self
已经被释放,则出现野指针访问