加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 百科 > 正文

Swift学习笔记之闭包

发布时间:2020-12-14 02:42:32 所属栏目:百科 来源:网络整理
导读:简介 (真的很简) 闭包的完整形态是这个样子的: { (parameters) - returnType in statements } 写在一行里就是这样: { (parameters) - (returnType) in statements} 形式 闭包以三种形式存在: 1.全局的函数都是闭包,它们有自己的名字,但是没有捕获任何值

简介 (真的很简)

闭包的完整形态是这个样子的:

{ (parameters) -> returnType in
        statements
    }

写在一行里就是这样:

{(parameters) -> (returnType) in statements}

形式

闭包以三种形式存在:

1.全局的函数都是闭包,它们有自己的名字,但是没有捕获任何值。
2.内嵌的函数都是闭包,它们有自己的名字,而且从包含他们的函数里捕获值。
3.闭包表达式都是闭包,它们没有自己的名字,通过轻量级的语法定义并且可以从上下文中捕获值。

捕获值

闭包可以捕获上下文的值,然后把它存储下来。至于存储的是引用还是拷贝,Swift 会决定捕获引用还是拷贝值,也会处理变量的内存管理操作。

下面这个例子可以说明很多问题:

func makeIncrementor(forIncrement amount: Int) -> () -> Int {
        var runningTotal = 0
        func incrementor() -> Int {
            runningTotal += amount
            return runningTotal
        }
        return incrementor
    }

    let incrementByTen = makeIncrementor(forIncrement: 10)
    incrementByTen()    // runningTotal = 10
    incrementByTen()    // runningTotal = 20
    incrementByTen()    // runningTotal = 30

    let incrementByTen2 = makeIncrementor(forIncrement: 10)
    incrementByTen2()    // runningTotal = 10

    let incrementByTen3 = incrementByTen
    incrementByTen()    // runningTotal = 40

因为 incrementByTen2 声明了一个全新的闭包,所以 runningTotal 并没有继续接着上面的计数。而 incrementByTen3incrementByTen 指向的是同一个闭包,所以 runningTotal 的值是累加的。

参数缩写

我们可以直接用 $0 $1 $2 这种来依次定义闭包的参数。比如 sorted 函数:

var reversed = sorted(["c","a","d","b"],{ $0 > $1 })   // d c b a

尾随闭包

我一直觉得闭包最后这个 }) 很难看,在 JS 中随处可见这种情况。如果闭包是函数的最后一个参数,Swift 提供了尾随闭包 (Trailing Closures) 解决这个审美问题:

// 以下是不使用尾随闭包进行函数调用
    someFunc({ // 闭包主体部分 })

    // 以下是使用尾随闭包进行函数调用
    someFunc() {
      // 闭包主体部分
    }

OK那么前面那个排序的可以用尾随闭包这么改写:

var reversed = sorted(["c","b"]) { $0 > $1 } // d c b a

如果除了闭包没有其他参数了,甚至可以把小括号也去掉。

还记得我们前面写的 mapreducefilter 三元大将吗?用尾随闭包可以让它们变得更好看。比如前面那个选出大于30的数字的 filter 就可以这样写:

var oldArray = [10,20,45,32]
    var filteredArray  = oldArray.filter{
        return $0 > 30
    }

    println(filteredArray) // [45,32]

变形

变形金刚神马的最有爱了。总结一下 closure 的变形大致有以下几种形态:

[1,2,3].map( { (i: Int) ->Int in return i * 2 } ) [1,2,3].map( { i in return i * 2 } ) [1,3].map( { i in i * 2 } ) [1,3].map( { $0 * 2 } ) [1,3].map() { $0 * 2 } [1,3].map { $0 * 2 }

对比

通过 UIView 的 animateWithDuration 方法对 block 和 closure 进行简单的对比。

block 版本:

    [UIView animateWithDuration:1 animations:^{
        // DO SOMETHING
    } completion:^(BOOL finished) {
        NSLog(@"OVER");
    }];

closure 版本:

    UIView.animateWithDuration(1,animations: { () in 
        // DO SOMETHING
        },completion:{(Bool)  in
            println("OVER")
        })

可以看到原来的 ^ 符号已经不复存在,取而代之的是加在参数和返回值后面的 in 。注意,如果有 in 的话,就算没有参数没有返回值也一定需要 () ,否则会报错。

总结

和 Objective-C 的 FuckingBlock 一样,Swift 出来之后 FuckingClosure 也就应运而生。总结了 Closure 的常用语法和格式:

// 作为变量
    var closureName: (parameterTypes) -> (returnType)

    // 作为可选类型的变量
    var closureName: ((parameterTypes) -> (returnType))? // 做为一个别名 typealias closureType = (parameterTypes) -> (returnType)

    // 作为函数的参数
    func({(parameterTypes) -> (returnType) in statements}) // 作为函数的参数 array.sort({ (item1: Int,item2: Int) -> Bool in return item1 < item2 }) // 作为函数的参数 - 隐含参数类型 array.sort({ (item1,item2) -> Bool in return item1 < item2 }) // 作为函数的参数 - 隐含返回类型 array.sort({ (item1,item2) in return item1 < item2 }) // 作为函数的参数 - 尾随闭包 array.sort { (item1,item2) in return item1 < item2 } // 作为函数的参数 - 通过数字表示参数 array.sort { return $0 < $1 } // 作为函数的参数 - 尾随闭包且隐含返回类型 array.sort { $0 < $1 } // 作为函数的参数 - 引用已存在的函数 array.sort(<)

References

  • Closures
  • Closure Expressions in Swift
  • Fucking Closure
  • Writing completion blocks with closures in Swift
  • Enough About Swift Closures to Choke a Horse
  • Functions and Closures in Swift
  • Swift How-To: Writing Trailing Closures

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读