Swift基础语法-闭包,闭包函数回调,尾随闭包,闭包捕获值,循环引用


本节知识点

  1. 闭包的基本概念
  2. 闭包基本使用
  3. 闭包表达式作为回调函数
  4. 闭包的多种写法(尾随闭包)
  5. 闭包表达式优化
  6. 闭包捕获值
  7. <a href="#闭包的循环引用(重点)">闭包的循环引用(重点)</a>

1. 闭包的基本概念

  • 闭包是一种类似于OC语言的block 匿名函数
  • block闭包都经常用于回调
  • 闭包表达式(匿名函数) 能够捕获上下文中的值
  • 闭包格式:
{
    (参数) -> 返回值类型 in
    执行语句
}
  • 闭包表达式的类型和函数的类型一样, 是参数加上返回值, 也就是in之前的部分
  • 语法: in关键字的目的是便于区分返回值和执行语句

2. 闭包基本使用

  • ** 有参数没有返回值写法**
let say0:(String) ->Void = {
    (name: String) in
    print("hi \(name)")
}
say0("cdh")
//输出结果:  hi cdh
  • 没有参数没有返回值写法
let say1:() ->Void = {
    () -> () in // 当没有参数也没有返回值是这行都可以省略
    print("hi cdh")
}
say1()
//输出结果:  hi cdh
  • 有参数有返回值
略; 按照格式, 结合函数的写法即可
  • 没有参数有返回值

3. 闭包表达式作为回调函数

3.1 block的用法回顾
  • 定义网络请求的类
@interface HttpTool : NSObject
- (void)loadRequest:(void (^)())callBackBlock;
@end

@implementation HttpTool
- (void)loadRequest:(void (^)())callBackBlock
{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"加载网络数据:%@", [NSThread currentThread]);

        dispatch_async(dispatch_get_main_queue(), ^{
            callBackBlock();
        });
    });
}
@end
  • 进行网络请求,请求到数据后利用block进行回调
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    [self.httpTool loadRequest:^{
        NSLog(@"主线程中,将数据回调.%@", [NSThread currentThread]);
    }];
}
  • block写法总结:
//block的写法:
//类型定义:
返回值(^block的名称)(block的参数)

//等号右边的值:
^(参数列表) {
    // 执行的代码
};
3.2 使用闭包代替block
  • 定义网络请求的类
class HttpTool: NSObject {

    func loadRequest(callBack : ()->()){
        dispatch_async(dispatch_get_global_queue(0, 0)) { () -> Void in
            print("加载数据", [NSThread.currentThread()])

             dispatch_async(dispatch_get_main_queue(), { () -> Void in
                callBack()
             })
        }
    }
}
  • 进行网络请求,请求到数据后利用闭包进行回调
    override func touchesBegan(touches: Set<UITouch>, withEvent event: UIEvent?) {
        // 网络请求
        httpTool.loadRequest ({ () -> () in
            print("回到主线程", NSThread.currentThread());
        })
    }
  • 闭包写法总结:
//闭包的写法:
// 类型:(形参列表)->(返回值)
// 技巧:初学者定义闭包类型,直接写()->().再填充参数和返回值

//值:
{
    (形参) -> 返回值类型 in
    // 执行代码
}
3.3 闭包的简写
  • 如果闭包没有参数,没有返回值.in和in之前的内容可以省略
    httpTool.loadRequest({
        print("回到主线程", NSThread.currentThread());
    })

4. 闭包的多种写法(尾随闭包)

  • 如果闭包是函数的最后一个参数,则可以将闭包写在()后面
  • 如果函数只有一个参数,并且这个参数是闭包,那么()可以不写
  • 这样可以提高阅读性. 称之为尾随闭包
    httpTool.loadRequest() {
        print("回到主线程", NSThread.currentThread());
    }
    // 开发中建议该写法
    httpTool.loadRequest {
        print("回到主线程", NSThread.currentThread());
    }

5. 闭包表达式优化

  • 类型优化, 由于函数中已经声明了闭包参数的类型, 所以传入的实参可以不用写类型
  • 返回值优化, 同理由于函数中已经声明了闭包的返回值类型, 所以传入的实参可以不用写类型
  • 参数优化, swift可以使用$索引的方式来访问闭包的参数, 默认从0开始
// 比较大小排序
func bubbleSort(inout array:[Int], cmp: (Int, Int) -> Int) {
    let count = array.count;
    // i = 1; i < count; i++ 推荐写成 i in 1..< count 
    // 在Swift3.0 也可能就只有后者这种写法
    for var i = 1; i < count; i++ {
        for var j = 0; j < (count - i); j++ {
            if cmp(array[j], array[j + 1]) == -1 {
                let temp = array[j]
                array[j] = array[j + 1]
                array[j + 1] = temp
            }
        }
    }
}
var arr:Array<Int> = [31, 13, 52, 84, 5]
print("排序前 \(arr)")
//输出结果: 排序前 [31, 13, 52, 84, 5]
bubbleSort(&arr){
//    (a , b) -> Int in
//    (a , b) in
    if $0 > $1{
        return 1;
    }else if $0 < $1 {
        return -1;
    }else {
        return 0;
    }
}
print("排序后 \(arr)")
//输出结果: 排序后 [84, 52, 31, 13, 5]
  • 如果只有一条语句可以省略return
let hehe = {
    "我是cdh"
}

6. 闭包捕获值

  • 被捕获的值会和与之对应的方法绑定在一起, 下一次需要用到这个方法这继续使用之前捕获到的值
  • 同一个方法中的变量, 不会被绑定到不同的方法变量(或常量中)
func getIncFunc() -> (Int) -> Int {
    var max = 10
    func incFunc(x :Int) ->Int{
        print("incFunc函数结束")
        max++  // ++ 这种写法将在 swift3.0 移除, 推荐写成 += 1
        return max + x
    }
    //当执行到这一句时incFunc 的参数 x 就应该被释放了
    //但是由于在内部函数中使用到了它, 所以它被捕获了
    //同理, 当执行完这一句时max变量就被释放了
    //但是由于在内部函数中使用到了它, 所以它被捕获了
    print("getIncFunc函数结束")
    return incFunc
}
let incFunc = getIncFunc()
print("---------")
print(incFunc(5))
print("---------")
print(incFunc(5))
//输出结果: 
//getIncFunc函数结束
//---------
//incFunc函数结束
//16
//---------
//incFunc函数结束
//17

let incFunc2 = getIncFunc() 
print(incFunc2(5))
//输出结果:
//getIncFunc函数结束
//incFunc函数结束
//16

7. <a name = "闭包的循环引用(重点)"></a>闭包的循环引用(重点)

  • 闭包形成的循环引用与 block 的循环引用是一样的原因导致. 如下闭包形成的循环引用的示例:
强引用 loadData函数的参数(闭包). png
loadData函数中参数(闭包)强引用 self(ViewController).png

//  ViewController.swift
//  03-13-闭包的循环引用
//
//  Created by chendehao on 16/3/2.
//  Copyright © 2016年 CDH. All rights reserved.
//

import UIKit

class ViewController: UIViewController {

    var httpTool : HttpTools?
    
    
    override func viewDidLoad() {
        super.viewDidLoad()
        
        httpTool = HttpTools()
    }
    
    override func touchesBegan(touches: Set<UITouch>, withEvent event: UIEvent?) {
        
        httpTool?.loadData(){ (jsonData) in
            print("在 viewController中拿到数据: \(jsonData)")
            self.view.backgroundColor = UIColor.grayColor()
            // self 强引用这类属性 httpTool, httpTool 中的函数 loadData 中的闭包参数在这里引用者 self (ViewController) 
            // 如果此次该闭包参数在 HttpTool 类中没有被强引用, 则此时将不形成循环引用,因为闭包是临时存储, 用完就被释放
            // 也就只有self (ViewController) 强引用这类属性 httpTool, 而 HttpTool 并没有强引用 self
            
            // 但是如果在 HttpTool 类中强引用了 loadData函数中的闭包这个参数, 则此时就会形成循环引用
            
        }
    }

    // 该对象被释放都会调用这个函数
    deinit{
        print("ViewController -- deinit")
    }
}
//  HttpTools.swift
//  03-13-闭包的简单使用
//
//  Created by chendehao on 16/3/2.
//  Copyright © 2016年 CDH. All rights reserved.
//

import UIKit

class HttpTools: NSObject {
    
    var finishedCallBack :((jsonData : String) -> ())?
    
    
    // 闭包的类型写法: (参数列表) -> (返回值类型)
    func loadData(finishedCallback : (jsonData : String) -> ())  {
        // 此处对函数中的闭包参数通过属性强引用
        self.finishedCallBack = finishedCallback
        
        dispatch_async(dispatch_get_global_queue(0, 0)) { () -> Void in
            print("正在发送网络请求: \(NSThread.currentThread())")
            
            dispatch_sync(dispatch_get_main_queue(), { 
                () -> Void in
                print("回调主线程, 见数据回传出去, \(NSThread.currentThread())")
                
                finishedCallback(jsonData: "json数据")
            })
        }
    }
}
  • 通过以上两段代码已经形成了强引用, 这样一来就会形成内存泄漏问题
7.1 解决循环引用
  • 未免出现以上的循环引用导致内存泄漏的问题, 有多种解决办法
  • 方法一: 删除对 loadData 函数中的闭包参数做强引用即可
// 删除这个即可
        // 此处对函数中的闭包参数通过属性强引用
        self.finishedCallBack = finishedCallback
  • 方法二(重点掌握): 将 self (ViewController) 使用弱引用关键之修饰
// 写法一: 定义一个 weak 修饰的新的临时变量
weak var weakSelf: ViewController? = self
httpTool?.loadData({ (jsonData) ->() in

    weakSelf?.view.backgroundColor = UIColor.orangeColor()
    print("在 viewController中拿到数据: \(jsonData)")
})
// 写法二: 直接使用 [weak self] 写在闭包的大括号中

// 尾随闭包 : 如果在调用方法时,该方法的最后一个参数是一个闭包.那么该闭包可以写成尾随闭包
// 尾随闭包写法一: (推荐将 [weak self]写在闭包的大括号中的写法)
httpTool?.loadData(){ [weak self] (jsonData) in
    print("在 viewController中拿到数据: \(jsonData)")
    self?.view.backgroundColor = UIColor.grayColor()
}

// 尾随闭包写法二:
httpTool?.loadData { [unowned self] (jsonData) in
    print("在 viewController中拿到数据: \(jsonData)")
    self.view.backgroundColor = UIColor.lightGrayColor()
}

// 注意 weak 和 unowned 的区别, 查看<<内存管理>>一文有详细说明

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 157,198评论 4 359
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 66,663评论 1 290
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 106,985评论 0 237
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 43,673评论 0 202
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 51,994评论 3 285
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,399评论 1 211
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,717评论 2 310
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,407评论 0 194
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,112评论 1 239
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,371评论 2 241
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 31,891评论 1 256
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,255评论 2 250
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 32,881评论 3 233
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,010评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,764评论 0 192
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,412评论 2 269
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,299评论 2 260

推荐阅读更多精彩内容