Block 与 闭包

2020-05-29  本文已影响0人  PlatoJobs

闭包类似于OC 中的 Block

  • 预先定义好的代码
  • 在需要时执行
  • 可以当作参数传递
  • 可以有返回值
  • 包含 self 时需要注意循环引用

1. Block

#import "GRController.h"
@implementation GRController
-(void)viewDidLoad{
    [super viewDidLoad];
    [self gr_loadData:^(NSString *str) {
        NSLog(@"回调代码 %@", str);
    }];
}

-(void)gr_loadData:(void (^)(NSString *str))finished {
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        NSLog(@"耗时操作");
        NSLog(@"%@",[NSThread currentThread]);
        dispatch_sync(dispatch_get_main_queue(), ^{
            NSLog(@"准备主线程回调");
            NSLog(@"%@",[NSThread currentThread]);
            //执行回调
            finished(@"AAAA");
        });
    });
}
@end

2. 闭包

Swift允许我们像字符串和整数一样使用函数。具体来说,你可以创建一个函数然后把它赋给一个变量,利用那个变量来调用函数。你甚至可以把函数作为参数传给另一个函数。

2.1 基本闭包

函数的这种用法被称为 闭包 。虽然工作机制差不多,写法上是有一些小差异的。
还是以打印信息为例:

let driving = {
  print("我要去开车")
}

上面的代码实际上创建了一个匿名的函数,并将这个函数赋给了 driving。之后你就可以把 driving() 当作一个常规的函数来用,就像这样:

driving()
2.2在闭包中接收参数

当你创建闭包的时候,它们并没有名字,也没有提供书写参数的地方。但这并不意味着它们不能接收参数,只不过它们接收参数的方式稍有不同:这些参数是被写在花括号里面的。

为了让一个闭包接收参数,你需要在花括号之后把这些参数列出来,然后跟上一个 in 关键字。这样就告诉Swift,闭包的主体是从哪里开始的。

举个例子,我们来创建一个闭包,接收一个叫 place 的字符串作为唯一的参数,就像这样:

let driving = { (place: String) in
  print("我要开车去 \(place)。")
}

函数和闭包的一个区别是运行闭包的时候你不会用到参数标签。因此,调用driving() 的时候,我们是这样写的:

driving("北京")
2.3 从闭包中返回值

闭包也能返回值,写法和闭包的参数类似:写在闭包内部, in 关键字前面。

还是以 driving() 闭包为例, 让它返回一个字符串。原来的函数是这样的:

let driving = { (place: String) in
  print("我要开车去  \(place)。")
}

改成返回字符串而不是直接打印那个字符串,需要 in 之前添加 -> String,然后像常规函数那样用到 return 关键字:

let drivingWithReturn = { (place: String) -> String in
  return "我要开车去 \(place)。"
}

现在我们运行这个闭包并且打印出它的返回值:

let message = drivingWithReturn("北京")
print(message)
2.4 闭包作为参数

既然闭包可以像字符串和整数一样使用,你就可以将它们传入函数。闭包作为参数的语法乍一看一看挺伤脑筋的,让我们慢慢来。

首先,还是基本的 driving() 闭包。

let driving = {
  print("我正在开车。")
}

如果我们打算把这个闭包传入一个函数,以便函数内部可以运行这个闭包。我们需要把函数的参数类型指定为 () -> Void。 它的意思是“不接收参数,并且返回 Void”。在Swift中,Void是什么也没有的意思。

好了,让我们来写一个 travel() 函数,接收不同类型的 traveling 动作, 并且在动作前后分别打印信息:

func travel(action: () -> Void) {
  print("我准备出发了。")
  action()
  print("我到达目的地了。")
}

现在可以用上 driving 闭包了,就像这样:

travel(action: driving)
2.5 拖尾闭包语法

如果一个函数的最后一个参数是闭包,Swift允许你采用一种被称为 “拖尾闭包语法” 的方式来调用这个闭包。你可以把闭包传入函数之后的花括号里,而不必像传入参数那样。

又用到我们的 travel() 函数了。它接收一个 action 闭包。闭包在两个 print() 调用之间执行:

func travel(action: () -> Void) {
  print("我准备出发了。")
  action()
  print("我到达目的地了。")
}

由于函数的最后一个参数是闭包,我们可以用拖尾闭包语法来调用 travel() 函数,就像这样:

travel() {
  print("我正在开车。")
}

实际上,由于函数没有别的参数了,我们还可以将圆括号完全移除:

travel {
  print("我正在开车。")
}

拖尾闭包语法在Swift中非常常见,所以你要适应它。
使用接收参数的闭包作为函数的参数

接下来要说到的闭包用法会有点复杂: 当你把闭包作为函数参数时,闭包本身也接收参数。

前面我们用 () -> Void 来表示“不接收参数,并且什么也不返回“,但实际上你可以在 () 里填上你任何想要闭包接收的参数类型。

再次用到 travel() 函数。函数只接收一个闭包作为参数,但这次闭包会接收一个字符串参数:

func travel(action: (String) -> Void) {
  print("我准备出发了。")
  action("北京")
  print("我到达目的地了。")
}

现在,当我们采用拖尾闭包语法调用 travel() 时,我们的闭包代码会要求接收一个字符串:

travel { (place: String) in
  print("我准备开车去\(place)。")
}
2.6 使用有返回值的闭包作为函数的参数

我们之前用 () -> Void 来表示“不接收参数,并且什么也不返回”。你可以把 Void 替换成任意的类型从而让闭包可以返回值。

还是 travel()函数,这次闭包会返回一个字符串。

func travel(action: (String) -> String) {
  print("我准备出发了。")
  let description = action("北京")
  print(description)
  print("我到达目的地了")
}

仍然用拖尾闭包语法来调用 travel(),闭包要求接收一个字符串并且返回一个字符串:

travel { (place: String) -> String in
  return "我要开车去\(place)。"
}
2.7 速记参数名

前面我们了构建 travel() 函数。它接收一个闭包作为参数,这个闭包本身接收一个参数并且返回一个字符串,它在两个 print() 调用之间运行。

代码如下:

func travel(action: (String) -> String) {
  print("我准备出发了。")
  let description = action("北京")
  print(description)
  print("我到达目的地了。")
}

我们可以像这样调用 travel():

travel { (place: String) -> String in
  return "我要开车去\(place)。"
}

不过,Swift知道提供给闭包的参数必须是一个字符串,所以调用的代码可以简写成这样:

travel { place -> String in
  return "我要开车去\(place)。"
}

Swfit也知道闭包必须返回一个字符串,于是进一步简写:

travel { place in
  return "我要开车去\(place)。"
}

由于这里的闭包只有一行代码,这行代码肯定是返回值的那行代码,因此Swift允许我们把 return 关键字也移除:

travel { place in
  "我要开车去\(place)。"
}

最后,Swift还提供一种速记语法,让你可以把代码变得更短。我们可以让Swift为闭包的参数自动提供一个名字,而不必自行写下 place in。这些自动生成的名字以$开头,然后跟着一个从0开始的整数,就像下面这样:

travel {
  "我要开车去\($0)。"
}
2.8 有多个参数的闭包

让我们把闭包这个概念一次讲透吧。接下来举一个接收两个参数的闭包的例子。

将 travel() 函数改造一下,不仅接收旅行目的地,也接收速度。闭包的类型会变成 (String, Int) -> String:

func travel(action: (String, Int) -> String) {
  print("我准备出发了。)
  let description = action("北京", 60)
  print(description)
  print("我到达目的地了。")
}

再一次用速记闭包参数名来调用函数。由于这次闭包有两个参数了,于是自动参数名分别是 0 和1:

travel {
  "我要开车去\($0),时速\($1)公里每小时。"
}

有些人可能不喜欢用速记参数名,因为它们的语义不是很清晰。你可以根据自己的喜好来决定是否采用它们。不过了解一下这个语法还是必要的,这样读到别人的代码时就不会感到困惑。

2.9 从函数中返回闭包

就如同你可以把闭包传入函数那样,你也可以从函数中返回闭包。

返回闭包的语法看起来有点绕,因为用了两次 ->:第一次用于指定函数的返回值,第二次用于指定闭包的返回值。

又又又要把 travel() 函数拉出来了。这次它不接收参数,但返回一个闭包。这个返回的闭包在用的时候必须传入一个字符串,但闭包本身没有返回值。

Swift代码长这样:

func travel() -> (String) -> Void {
  return {
    print("我要动身去\($0)")
  }
}

接下来我们通过调用 travel() 拿到闭包,然后作为函数来调用:

let result = travel()
result("北京")

留意下面的代码,它是直接调用 travel() 的返回值。这个写法虽然在语法上完全没问题,但是可读性较差,建议尽量不要这样写。

let result2 = travel()("北京")
2.10 捕获变量

如果你想要使用闭包之外的对象,Swift会为你“捕捉”它们,并把它们和闭包一同存储,以便外部作用域已经失效的情况下闭包内部还可以使用它们。

最后一次用到 travel() 函数,它返回一个闭包,这个闭包接收字符串作为唯一的参数并且什么也不返回:

func travel() -> (String) -> Void {
  return {
    print("我准备去\($0)")
  }
}

调用 travel() 拿到闭包,然后自由使用:

let result = travel()
result("北京")

闭包捕获变量可以发生在什么情况下呢?举个例子,当 travel() 函数内创建了一个变量,这个变量需要在闭包里面用到,那么这个变量就会被闭包捕获。比如,我们想知道闭包被调用的次数:

func travel() -> (String) -> Void {
  var counter = 1
  return {
    print("第\(counter)次,我将前往\($0)")
    counter += 1
  }
}

尽管 counter 变量是在 travel() 里被创建的,它被闭包捕获,因而会在闭包内部存续。

当我们多次调用 result("北京"),计数器会持续增加:

result("北京")
result("北京")
result("北京")

总结

上一篇 下一篇

猜你喜欢

热点阅读