Swift 函数 & 闭包

Swift 函数

Swift 函数用来完成特定任务的独立的代码块。
Swift使用一个统一的语法来表示简单的C语言风格的函数到复杂的Objective-C语言风格的方法。

  • 函数声明:告诉编译器函数的名字,返回类型及参数
  • 函数定义:提供了函数的实体
    Swift 函数包含了参数类型及返回值类型
函数定义

Swift 定义函数使用关键字 func
定义函数的时候,可以指定一个或多个输入参数和一个返回值类型
每个函数都有一个函数名来描述该函数的功能。通过函数名以及队形类型的参数值来调用这个函数。函数的参数传递的顺序必须与参数列表相同。
函数的实参传递的顺序必须与形参列表相同, ->后定义函数的返回值类型。

函数语法
func funcname(形参)  ->  returntype{
Statement1
Statement2
……
StatementN
return parameters
}
实例

一下定义了一个函数名为runoob的函数,形参的数据类型为String,返回值也为String:

func runoob(site: String) -> String {
      return(site)
}
print(ruboob(site:"www.baidu.com"))

以上程序执行输出的结果为:

www.baidu.com
函数参数

函数可以接受一个或者多个参数,这些参数被包含在函数的括号中,以逗号分隔。以下实例向函数 mj_printAllName 传递 firstNamelastName

 func mj_printAllName(firstName: String, lastName: String) -> String {

        return firstName + lastName  
 }
不带参数的函数
func funcName() -> datatype {
      return datatype 
}
元组作为函数返回值

函数返回值类型可以是 字符串、整型、浮点型等
元组与数组类似,不同的是,元组的元素可以是任意类型,使用的是圆括号
你可以用元组(tuple)类型让多个值作为一个复合值从函数返回
定义了一个名为 minMax(_:) 的函数作用是在一个Int数组中找出最大值与最小值

func minMax(array: [Int]) -> (min: Int, max: Int) {
    var currentMin = array[0]
    var currentMax = array[0]
    for value in array[1..<array.count] {
        if value < currentMin {
            currentMin = value
        } else if value > currentMax {
            currentMax = value
        }
    }
    return (currentMin, currentMax)
}

let bounds = minMax(array: [8, -6, 2, 109, 3, 71])
print("最小值为 \(bounds.min) ,最大值为 \(bounds.max)")

minMax(_:)函数返回一个包含两个Int值的元组,这些值被标记为min和max,以便查询函数的返回值时可以通过名字访问它们。
以上程序输出:

最小值为 -6 ,最大值为 109

如果你不确定返回的元组一定不为nil,那么你可以返回一个可选的元组类型。
你可以通过在元组类型的右括号后放置一个问号来定义一个可选元组,例如 (Int, Int)?或 (String,Int,Bool)?

注意:可选元组如 (Int,Int)? 与元组包含可选类型如 (Int?, Int?)是不同的,可选的元组类型,整个元组都是可选的,而不是只元组中的每个元素值。

前面minMax(_ :)函数反悔了一个包含两个Int值的元组。但是函数不会对传入的数组进行任何安全检查,如果array参数是一个空数组,如上定义的minMax(:)在访问array[0]时就触发一个运行时错误。为了安全处理这个 “空数组”问题,将minMax(:)函数改写为使用可选元组返回类型,并且当数组为空时返回nil:

func minMax(array: [Int]) -> (min: Int, max: Int)? {
    if array.isEmpty { return nil }
    var currentMin = array[0]
    var currentMax = array[0]
    for value in array[1..<array.count] {
        if value < currentMin {
            currentMin = value
        } else if value > currentMax {
            currentMax = value
        }
    }
    return (currentMin, currentMax)
}
if let bounds = minMax(array: [8, -6, 2, 109, 3, 71]) {
    print("最小值为 \(bounds.min),组大值为 \(bounds.max)")
}

以上执行输出结果为:

最小值为 -6,组大值为 109
函数参数名称

函数参数都有一个外部参数和一个局部参数名。

局部参数

局部参数名在函数的实现内部使用。

func sample(number: Int) {
       print(number)
}

以上实例中 number为局部参数,只能在函数体内使用。

外部参数名

你可以在局部参数名前指定外部参数名,中间以空格分隔,外部参数名在用于函数调用时传递给函数的参数。
如下定义了以下两个函数参数并调用它:

 func mj_funcOutterAndInnerParameterTest(publicParameter1 a1 :Int, 
 publicParameter2 a2 :Int) -> Int {
        
        return a1 + a2;
    }

注意:如果提供了外部参数名,那么函数在被调用时,必须使用外部参数名。

可变参数

可变参数可以接受零个或者多个值。函数调用时,你可以用可变参数来指定函数参数,其数量是不确定的。可变参数通过在变量类型名后加入(…)的方式定义。

 func mj_testMutableParameter<paramters>(members:paramters...) {
        
        for item in members {
            print(item)
        }
    }
常量、变量及 I/O 参数

一般默认在函数中定义的参数,可以都是常量参数,也就是这个参数你只可以查询使用不能改变它的值。如果声明一个变量参数,可以在参数定义钱加 inout 关键字,专业那个就可以改变这个参数的值了。

func getName(_ name: inout String)…………

此时这 name 值可以在函数中改变
一般默认的参数传递都是传值调用,而不是传引用。所以传入的参数在函数内改变,并不影响原来的那个参数。传入的只是这个参数的副本。当传入的参数作为输入输出参数时,需要在参数明前加 & 符,表示这个值可以被函数修改。

  func mj_testMutableParameter<paramters>(members:paramters...) {
        
        for item in members {
            print(item)
        }
    }

var a = 10
var b = 20
mj_testParameterInout(&a, &b)
print("a is \(a), b is \(b)")

以上程序输出的结果是:

a is 20, b is 10
函数类型及使用

每个函数都有特定的函数类型,由函数的参数类型和返回类型组成。

func inputs(no1: Int, no2: Int) -> Int {
     return no1/no2
}

inputs 函数类型有两个Int 型的参数(no1、no2)并返回一个 Int 型的值。

使用函数类型

在 Switf 中使用函数类型就像使用其他类型一样。例如,你可以定义一个类型为函数的常量或变量,并适当的函数赋值给它。

var addition: (Int, Int) -> Int = sum

解析:“定义一个叫addition的变量,参数与返回值类型均为 Int,并让这个新变量指向 sum 函数”。sum 和 addition 有同样的类型,所以以上操作是合法的。
现在,你可以使用addition 来调用被赋值的函数了。

func sum(a: Int, b: Int) -> Int {
   return a + b
}
var addition: (Int, Int) -> Int = sum
print("输出结果: \(addition(40, 89))")

输出的结果为:

输出结果: 129
函数类型作为参数类型,函数类型作为返回类型

我们可以将函数作为参数传递给另一个参数:

func sum(a: Int, b: Int) -> Int {
    return a + b
}
var addition: (Int, Int) -> Int = sum
print("输出结果: \(addition(40, 89))")

func another(addition: (Int, Int) -> Int, a: Int, b: Int) {
    print("输出结果: \(addition(a, b))")
}
another(addition: sum, a: 10, b: 20)

输出结果为:

输出结果: 129
输出结果: 30
函数嵌套

函数嵌套指的是函数内定义一个新的函数,外部的函数可以调用函数内定义的函数。

func calcDecrement(forDecrement total: Int) -> () -> Int {
   var overallDecrement = 0
   func decrementer() -> Int {
      overallDecrement -= total
      return overallDecrement
   }
   return decrementer
}
let decrem = calcDecrement(forDecrement: 30)
print(decrem())

以上程序执行输出结果为:

-30

闭包

闭包(closures)是自包含的功能代码块,可以在代码中使用或者用来作为参数传值。Swift 中闭包与 COC中代码块(block)以及其他一些编程中的 匿名函数相似。
全局函数和嵌套函数其实就是特殊的闭包。
闭包的行驶:

全局函数 嵌套函数 闭包表达式
有名字但不能捕获任何值 有名字,也能捕获封闭函数内的值。 无名闭包,使用轻量级语法,可以根据上下文环境捕获值

Swift中闭包有很多优化的地方:

  1. 根据上下文推断参数和返回值
  2. 从单行表达式闭包中隐式返回(也就是闭包体只有一行代码,可以省略 return)
  3. 可以使用简化参数名,如0,1(从0开始,表示第 i 个参数…)
  4. 提供了尾随闭包语法(Trailing closure syntax)
语法

以下定义了一个接收并返回指定类型的闭包语法:

{(parameter) -> return type in 
  statements
}
let sutdentName = {print("Swift 闭包实例")}

stutentName()

上面程序输出结果为:

Switf 闭包实例

以下闭包形式接收两个参数并返回布尔值:

{(Int, Int) -> Bool  in 
   Statement1
   Statement2
    ……
  Statement N
}
let divide = {(val1: Int, val2: Int) -> Int in 
   return val1 / val2 
}
let result = divide(200, 20)
print (result)

以上程序执行的结果是:

10
闭包表达式

闭包表达式是一种利用简洁语法构建内联闭包的方式。闭包表达式提供了一些语法优化,是得撰写闭包变得简单明了。

sorted方法

Swift 标准库提供了名为:sorted(by:) 的方法,会根据您提供的用于排序的闭包函数将已知的类型数组中的值进行排序。排序完成后,sorted(by:) 方法返回一个与原始数组大小相同,包含同类型元素且元素已正确排序的新数组。原数组不会被 sorted(by:) 方法修改。
sorted(by:) 方法需要传入两个参数:

  1. 已知类型的数组
  2. 闭包函数,该闭包函数需要传入与数组元素类型相同的值,并且返回一个布尔类型值来表明当前排序结束后传入的第一个参数排在第二个参数前面还是后面。如果第一个参数值出现在第二个参数前面,排序闭包函数需要返回 true 反之返回 false
let names = ["AT", "AE", "D", "S", "BE"]

// 使用普通函数(或内嵌函数)提供排序功能,闭包函数类型需为(String, String) -> Bool。
func backwards(s1: String, s2: String) -> Bool {
    return s1 > s2
}
var reversed = names.sorted(by: backwards)

print(reversed)

以上程序执行输出结果为:

["S", "D", "BE", "AT", "AE"]

如果第一个字符串 (s1) 大于第二个字符串 (s2),backwards函数返回true,表示在新的数组中s1应该出现在s2前。 对于字符串中的字符来说,"大于" 表示 "按照字母顺序较晚出现"。 这意味着字母"B"大于字母"A",字符串"S"大于字符串"D"。 其将进行字母逆序排序,"AT"将会排在"AE"之前。

参数名称缩写

Swift 自动为内联函数提供了参数名称的缩写功能直接通过 $0,$1,$2 来顺序调用闭包的参数。

let names = ["AT", "AE", "D", "S", "BE"]

var reversed = names.sorted( by: { $0 > $1 } )
print(reversed)

$0$1表示闭包中第一个和第二个String类型的参数。
以上程序执行输出结果为:

["S", "D", "BE", "AT", "AE"]

如果你在闭包表达式中使用参数名称缩写, 您可以在闭包参数列表中省略对其定义, 并且对应参数名称缩写的类型会通过函数类型进行推断。in 关键字同样也可以被省略.

运算符函数

实际上还有一种更简短的方式来撰写上面例子中的闭包表达式。
Swift 的 String类型定义了关于大于号(>)的字符串实现,其作为一个函数接受两个 String类型的参数并返回Bool类型的值。这正好与sort(_:)方法的第二个参数需要的函数类型相符合。因此,您可以简单的传递一个大于号,Swift可以自动推断出您用使用大于号的字符串函数实现:

let names = ["AT", "AE", "D", "S", "BE"]
var reversed = names.sorted(by: >)
print(reversed)

以上程序执行输出结果为:

["S", "D", "BE", "AT", "AE"]
尾随闭包

尾随闭包是一个书写在函数括号之后的闭包表达式,函数支持将其作为最后一个参数调用。

func someFunctionThatTakesAClosure(closure: () -> Void) {
    // 函数体部分
}

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

// 以下是使用尾随闭包进行函数调用
someFunctionThatTakesAClosure() {
  // 闭包主体部分
}
let names = ["AT", "AE", "D", "S", "BE"]

//尾随闭包
var reversed = names.sorted() { $0 > $1 }
print(reversed)

sort()后的 {$0 > $1}为尾随闭包。以上程序输出的结果为:

["S", "D", "BE", "AT", "AE"]

注意:如果函数只需要闭包表达式一个参数,当你使用尾随闭包时,您甚至可以把 ()省略掉。

reversed = names.sorted {$0 > $1}
捕获值

闭包可以在其定义的上下文忠捕获常量或变量
及时定义这些常量和变量的原域已经不存在,闭包仍然可以在闭包函数体内引用和修改这些值。
Swift 最简单的闭包行驶就是嵌套函数,也就是定义在其他函数体内的函数
嵌套函数可以捕获其外部函数所有的参数以及定义的常量和变量

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

一个函数makeIncrementor,他有一个 Int 型的参数 amout,并且他有一个外部参数名字 forIncrement,意味着你调用的时候必须使用这个外部名字。返回值是一个 () -> Int 的函数。函数内声明了变量 runningTotal 和一个函数 incrementor。incrementor 函数并没有捕获任何参数,但是在函数体内访问了runningtTotal 和 amout变量。这是因为其通过捕获在包含它的函数体内已经存在的runningTotal和amout变量实现
由于没有修改amount变量,incrementor实际上捕获并存储了该变量的一个副本,而该副本随着incrementor一统存储。所以我们调用这个函数时会累加:

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

let incrementByTen = makeIncrementor(forIncrement: 10)

// 返回的值为10
print(incrementByTen())

// 返回的值为20
print(incrementByTen())

// 返回的值为30
print(incrementByTen())
闭包时引用类型

上面的例子中 incrementByTen是常量,但是这些常量指向的闭包仍然可以增加其捕获的变量值。
这是因为函数和闭包都有引用类型
无论那您将函数/闭包赋值给一个常量还是变量,您实际上都是将常量/变量的值设置为对应函数/闭包引用。上面的例子中,incrementByTen指向闭包的引用是一个常量,而并非闭包内容本身。
这也意味着如果您将闭包赋值给了量给不同的常量/变量,两个值都会指向同一个闭包:

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

let incrementByTen = makeIncrementor(forIncrement: 10)

// 返回的值为10
incrementByTen()

// 返回的值为20
incrementByTen()

// 返回的值为30
incrementByTen()

// 返回的值为40
incrementByTen()

let alsoIncrementByTen = incrementByTen

// 返回的值也为50
print(alsoIncrementByTen())

以上程序执行输出结果为:

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

推荐阅读更多精彩内容