Swift2.0语言教程之闭包

Swift2.0语言闭包

闭包是自包含的函数代码块,可以在代码中被传递和使用。Swift中的闭包与CObjective-C中的代码块(blocks)以及其他一些编程语言中的Lambda函数比较相似。在本章中所讲的函数其实就是特殊的闭包。本节主要讲解关于闭包的基本使用。

Swift2.0语言闭包表达式

闭包表达式是一种利用简洁语法构建内联(内联类似与C语言中的宏定义)闭包的方式。以下这个代码是对两个字符串的比较。

  • import Foundation

  • //判断两个字符串的大小

  • func compare(s1: String,s2: String) -> Bool {

  • return s1 > s2

  • }

  • let str1="Hello"

  • let str2="Swift"

  • if compare(str1,s2: str2) {

  • print("str1大于str2")

  • }else{

  • print("str1小于str2")

  • }

运行结果如下:

  • str1小于str2

在此代码中,当调用函数compare()时,会将str1str2传递到给函数在定义时的参数s1s2中,然后进行比较。在此代码中可以看到,在compare()函数中执行了一行代码,就是判断并返回,显的此函数相当冗长。此时就可以使用闭包表达式使用代码变得更好。首先,来看一下闭包表达式(闭包函数)的语法形式。

    {(参数列表)->返回值类型in
  • 语句

  • }

其中,参数可以是常量、变量和输入-输出参数,但没有默认值。开发者也可以在参数列表的最后使用可变参数。而元组也可以作为参数和返回值。关键字in表示闭包的参数和返回值类型定义已经完成,闭包函数体即将开始。

1.无参形式的闭包表达式

无参形式的闭包表达式语法形式如下:

    {()->返回值类型 它定义的语法形式如下:
      let/var
    闭包表达式常量名称/闭包表达式变量名称/=无参形式的闭包表达式

它调用的语法形式如下:

  • 闭包表达式常量名称()

【示例7-27】以下将使用闭包表达式实现字符串"aaaa"的输出。代码如下:

    //输出字符串"aaaa"
  • var str={() in

  • print("aaaa")

  • }

  • str()//调用的调用形式

  • aaaa

  • 2.具有参数的闭包表达式

    具有参数的闭包表达式有两种形式一种是最常用的只有一个参数的闭包表达式;一种是具有多个参数的闭包表达式。以下是对这两种具有参数的闭包表达式的详细讲解。

    1具有一个参数的闭包表达式

    具有一个参数的闭包表达式的语法形式如下:

      {(参数名:数据类型/=具有一个参数的闭包表达式

    它的调用形式如下:

      【示例7-28】以下将使用闭包表达式输出指定字符串。代码如下:
        //输出指定的字符串
      • var str={(str:String) in

      • print(str)

      • }

      • str("Hello")

      Hello

    2具有多个参数的闭包表达式

    具有多个参数的闭包表达式的语法形式如下:

      1:数据类型,参数名2:数据类型/=具有多个参数的闭包表达式

    【示例7-29】使用闭包实现对两个任意数的求和计算,代码如下:

      //实现求两个数的和
    • var reversed = {(s1: Int,s2: Int) -> Int in

    • var sum=s1+s2

    • return sum

    • }

    • print(reversed(10,20))

    30

    其实闭包表达式最长用在其他的函数中,并不是单独的去使用它。

    【示例7-30】以下代码将闭包表达式作为函数的一部分,来实现在判断在数组中是否有大于500或者40的元素。代码如下:

      //定义函数
    • func copare(arr:[Int],value:Int,cb:(Num:Int,Value:Int)->Bool)->Bool{

    • //遍历数组

    • for item in arr{

    • //判断闭是否为真

    • if(cb(Num: item,Value: value)){

    • return true

    • }

    • }

    • return false

    • }

    • var array = [20,80,100,50,20]

    • //使用闭包判断是否在数组中有大于500的元素

    • var v1=copare(array,value: 500,cb: {(num:Int,value:Int)->Bool in

    • return num>value

    • })

    • //判断结果并输出·

    • if v1==true {

    • print("数组array中有比500大的元素")

    • }else{

    • array中没有比}

    • 40的元素

    • var v2=copare(array,value: 40,sans-serif; line-height:1.5">//判断结果并输出

    • if v2==true {

    • 40大的元素}

    在此代码中,当调用copare()函数时,会将参数array40以及闭包表达式传递到函数定义的参数中,在copare()函数中,遍历数组中元素的时候,又会去调用闭包表达式。运行结果如下:

    • 数组500大的元素

    • 数组40大的元素

    Swift2.0语言使用闭包表达式的注意事项

    在使用闭包表达式时需要注意以下几点(以下都是以示例7-30来说明的):

    1.推断类型

    copare()函数的第三个参数是闭包表达式,它的类型为(num:Int,value:Int)->Bool,由于Swift可以推断其参数和返回值的类型,所以->和围绕在参数周围的括号可以省略,如以下的代码:

    • var v1=copare(array,value:500,cb:{(num,value) in

    • return num>value

    • })

    2.省略return

    单行表达式闭包可以通过隐藏return关键字来隐式返回单行表达式的结果,可以将上面的例子进行修改:

      num>value

    • })

    3.简写参数名

    Swift为内联函数提供了参数名缩写功能,开发者可以通过$0$1$2来顺序的调用闭包的参数。如果在闭包表达式中使用参数名称缩写,可以在闭包参数列表中省略对其的定义,并且对应参数名称缩写的类型会通过函数类型进行推断。in关键字也同样可以被省略,因为此时闭包表达式完全由闭包函数体构成,将上面的例子进行修改:

      $0 > $1

    • })

    4.写在一行

    当闭包的函数体部分很短时可以将其写在一行上面,如以下代码:

      5.运算符函数

      SwiftString类型定义了关于大于号(>)的字符串实现,其作为一个函数接受两个String类型的参数并返回Bool类型的值。而这正好与以上代码sort函数的第二个参数需要的函数类型相符合。因此,可以简单地传递一个大于号,Swift可以自动推断出您想使用大于号的字符串函数实现:

        Swift 1.2中使用闭包表达式需要注意以下三点:
        • q有单返回语句的闭包,现在类型检查时以单表达式闭包处理。

        • q匿名的且含有非空返回类型的单表达式,现在可以用在void上下文中。

        • q多表达式的闭包类型的情况,可能无法被类型推断出来。

        Swift2.0语言Trailing闭包

        如果开发者需要将一个很长的闭包表达式作为最后一个参数传递给函数,可以使用Trailing闭包,它可以增强函数的可读性。Trailing闭包的一般形式如下:

        • func someFunctionThatTakesAClosure(closure: () -> ()) {

        • //函数主体部分

        • }

        • //以下不是使用trailing闭包进行的函数调用

        • someFunctionThatTakesAClosure({

        • //闭包主体部分

        • })

        • //以下是使用trailing闭包进行的函数调用

        • someFunctionThatTakesAClosure() {

        • //闭包主体部分

        • 注意:trailing闭包是一个写在函数括号之后的闭包表达式,函数支持将其作为最后一个参数调用。示例7-30中代码也可以写为Trailing闭包,代码如下:

          • import Foundation

          • func copare(arr:[Int],Value:Int)->Bool)->Bool{

          • for item in arr{

          • if(cb(Num: item,20]

          • }

          • ifv1==true {

          • ")

          • }else{

          • ")

          • }

          Trailing闭包一般使用在当闭包很长以至于不能在一行进行编写的代码中。如以下的例子就使用了Trailing闭包,实现将数字改为英文的功能。代码如下:

            //创建字典
        • let digitNames = [

        • 0: "Zero",1: "One",2: "Two",3: "Three",4: "Four",5: "Five",6: "Six",7: "Seven",8: "Eight",9: "Nine"

        • ]

        • //创建数组

        • let numbers = [521,52,1,13,14]

        • trailing闭包进行的函数调用,实现将数字转为英文

        • let strings = numbers.map {

        • (var number) -> String in

        • var output = ""

        • while number > 0 {

        • output = digitNames[number % 10]! + output

        • number /= 10

        • }

        • return output

        • //遍历并输出

        • for index in strings{

        • print(index)

        • }

        运行结果如下所示:

          FiveTwoOne
        • FiveTwo

        • One

        • OneThree

        • OneFour

        注意:在此代码中使用到了函数map(),它的功能是返回一个新的序列。其语法形式如下:

          map(序列闭包表达式)

        其中,如果闭包表达式适用于序列中的所有元素,就会返回一个新的序列。但是在本示例中由于闭包中的内容比较多,就将它使用了Trailing闭包的形式。

        Swift2.0语言捕获值

        闭包可以在其定义的上下文中捕获常量或变量。以下就使用incrementor()函数从上下文种对值runningTotalamount进行捕获。代码如下:

        • import Foundation

        • func makeIncrementor(forIncrement amount: Int) -> () -> Int {

        • var runningTotal = 0

        • //定义函数incrementor(),实现runningTotal的增加

        • func incrementor() -> Int {

        • runningTotal += amount

        • return runningTotal

        • return incrementor

        • //赋值

        • var a = makeIncrementor(forIncrement: 10)

        • //输出

        • print("输出a的增量")

        • print(a())

        • print(a())

        • print(a())

        • var b = makeIncrementor(forIncrement: 5)

        • //赋值,输出

        • b的增量")

        • print(b())

        • print(b())

        • print(b())

      • 输出a的增量

      • 10

      • 20

      • 30

      • 输出b的增量

      • 5

      • 15

      本文选自:Swift2.0语言快速入门v3.0大学霸内部资料,转载请注明出处,尊重技术尊重IT人!

    Swift2.0语言教程之闭包的更多相关文章

    1. ios – Swift中的UIView动画不起作用,错误的参数错误

      我正在尝试制作动画并使用下面的代码.我得到“无法使用类型’的参数列表调用’animateWithDuration'(FloatLiteralConvertible,延迟:FloatLiteralConvertible,选项:UIViewAnimationoptions,动画:()–>()–>$T4,完成:(Bool)–>(Bool)–>$T5)’“错误.这意味着我使用了错误的参数.我错了.请

    2. ios – 使用捕获列表中的无主内容导致崩溃,即使块本身也不会执行

      欣赏有关如何调试此内容的任何提示或有关导致崩溃的原因的解释……

    3. ios – Swift传递封闭与Params

      目前我传递一个闭包作为一个对象的属性,该对象不接受参数并且没有返回值,如下所示:到目前为止,这工作得很好.我希望能够在设置此闭包时传入一个参数,以便在MyClass的实例中使用.我正在寻找下面的SOMETHING,虽然我确定语法不正确:我如何将参数传递给可以在MyClass中使用的闭包–即可以在属性本身的didSet部分内使用的值,如第二个示例中所示?

    4. ios – 来自UIAlertController的self.navigationController?.popViewControllerAnimated

      我是新手,但我想我已经掌握了它.这让我的进步很难过.我想要做的是当我们无法找到他的查询的相关数据时向用户抛出错误消息,然后继续将他带回到之前的ViewController.但是,我在这方面遇到了麻烦.在我添加操作的行上,我收到以下错误:’UIViewController?’不是Void的子类型我该怎么做呢?

    5. ios – 如何添加@noescape注释到可选闭包

      我的功能有这个签名:而现在我想让自己在给定的关闭中逃脱自我.但是当我尝试这个:编译器抱怨:是否可以在可选参数中使用它?

    6. ios – Swift闭包为AnyObject

      如何将()–>()转换为AnyObject?我试图将它转换为:处理程序为AnyObject,但它给我一个错误说:()–>()不符合协议’AnyObject’解决方法HowcanIcast()->()intoAnyObject?

    7. ios – 在swift中将捕获列表正确放置在嵌套闭包中

      在Swift中为哪些嵌套闭包定义捕获的引用?如果[weakself]被捕获在只有内部最后面的闭包,GCD将保留ExampleDataSource,直到块完成执行,这就解释了为什么调试看起来像这样:同样的事情会发生,如果没有捕获列表被包括,我们从来没有可选地解开自己,尽管编译器,确实试图警告你!

    8. ios – 两个类,回调和单元测试

      而且,我不觉得这样的行为会很容易测试?

    9. 寒城攻略:Listo 教你 25 天学会 Swift 语言 - 18 Automatic Reference Counting

      //每个Person类的实例拥有一个被初始化为nil的apartment可选属性,因为一个人并不一定拥有一座公寓)}}classApartment{letnumber:Intinit{number=number}tenant:Person1?//声明Apartment2类中的tenant属性为弱引用varPin:Person2?鉴于这种关系,Customer类有一个可选类型属性card,而CreditCard类的customer属性则是非可选类型的classCustomer{varcard:CreditC

    10. 【Swift初见】Swift闭包

      闭包是功能性自包含模块,可以在代码中被传递和使用。闭包表达式提供了一些语法优化,使得撰写闭包变得简单明了。闭包的函数体部分由in关键字引入,in后面就是代表该闭包的函数体部分。swift函数有类型推断的功能,排序闭包又是作为函数的参数传入的,那么swift可以推断出它的参数和返回值类型,那么sorted函数可以改写为:我们可以看出得出的结果还是一样的。单行表达式闭包可以通过隐藏return关键字来隐式返回表达式的结果。

    随机推荐

    1. Swift UITextField,UITextView,UISegmentedControl,UISwitch

      下面我们通过一个demo来简单的实现下这些控件的功能.首先,我们拖将这几个控件拖到storyboard,并关联上相应的属性和动作.如图:关联上属性和动作后,看看实现的代码:

    2. swift UISlider,UIStepper

      我们用两个label来显示slider和stepper的值.再用张图片来显示改变stepper值的效果.首先,这三个控件需要全局变量声明如下然后,我们对所有的控件做个简单的布局:最后,当slider的值改变时,我们用一个label来显示值的变化,同样,用另一个label来显示stepper值的变化,并改变图片的大小:实现效果如下:

    3. preferredFontForTextStyle字体设置之更改

      即:

    4. Swift没有异常处理,遇到功能性错误怎么办?

      本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容,请发送邮件至dio@foxmail.com举报,一经查实,本站将立刻删除。

    5. 字典实战和UIKit初探

      ios中数组和字典的应用Applicationschedule类别子项类别名称优先级数据包contactsentertainment接触UIKit学习用Swift调用CocoaTouchimportUIKitletcolors=[]varbackView=UIView(frame:CGRectMake(0.0,0.0,320.0,CGFloat(colors.count*50)))backView

    6. swift语言IOS8开发战记21 Core Data2

      上一话中我们简单地介绍了一些coredata的基本知识,这一话我们通过编程来实现coredata的使用。还记得我们在coredata中定义的那个Model么,上面这段代码会加载这个Model。定义完方法之后,我们对coredata的准备都已经完成了。最后强调一点,coredata并不是数据库,它只是一个框架,协助我们进行数据库操作,它并不关心我们把数据存到哪里。

    7. swift语言IOS8开发战记22 Core Data3

      上一话我们定义了与coredata有关的变量和方法,做足了准备工作,这一话我们来试试能不能成功。首先打开上一话中生成的Info类,在其中引用头文件的地方添加一个@objc,不然后面会报错,我也不知道为什么。

    8. swift实战小程序1天气预报

      在有一定swift基础的情况下,让我们来做一些小程序练练手,今天来试试做一个简单地天气预报。然后在btnpressed方法中依旧增加loadWeather方法.在loadWeather方法中加上信息的显示语句:运行一下看看效果,如图:虽然显示出来了,但是我们的text是可编辑状态的,在storyboard中勾选Editable,再次运行:大功告成,而且现在每次单击按钮,就会重新请求天气情况,大家也来试试吧。

    9. 【iOS学习01】swift ? and !  的学习

      如果不初始化就会报错。

    10. swift语言IOS8开发战记23 Core Data4

      接着我们需要把我们的Rest类变成一个被coredata管理的类,点开Rest类,作如下修改:关键字@NSManaged的作用是与实体中对应的属性通信,BinaryData对应的类型是NSData,CoreData没有布尔属性,只能用0和1来区分。进行如下操作,输入类名:建立好之后因为我们之前写的代码有些地方并不适用于coredata,所以编译器会报错,现在来一一解决。

    返回
    顶部