1、函数的基本概念

  • 函数是一个独立的代码块,用来执行特定的任务。通过给函数一个名字来定义它的功能,并且在需要的时候,通过这个名字来“调用”函数执行它的任务
  • Swift 统一的函数语法十分灵活,可以表达从简单的无形式参数的 C 风格函数到复杂的每一个形式参数都带有局部和外部形式参数名的 Objective-C 风格方法的任何内容。形式参数能提供一个默认的值来简化函数的调用,也可以被当作输入输出形式参数被传递,它在函数执行完成时修改传递来的变 量
  • Swift 中的每一个函数都有类型,由函数的形式参数类型和返回类型组成。你可以像 Swift 中其他类 型那样来使用它,这使得你能够方便的将一个函数当作一个形式参数传递到另外的一个函数中,也可以在一个函数中返回另一个函数。函数同时也可以写在其他函数内部来在内嵌范围封装有用的功能

2、函数定义的几种类型

1> 无形式参数的函数

  • 函数没有要求必须输入一个参数,可以没有形式参数
  • 函数的定义仍然需要在名字后边加一个圆括号,即使它不接受形式参数也得这样做。当函数被调用的时候也要在函数的名字后边加一个空的圆括号
//无形式参数
func sayHelloWorld() -> String {
    return "hello world"
}
sayHelloWorld()  //调用也需要带圆括号

2> 多形式参数的函数

  • 函数可以输入多个形式参数,可以写在函数后边的圆括号内,用逗号分隔
//多形式参数
func addTwoNumbers(num1: Int, num2: Int) -> Int {
    return num1 + num2
}
addTwoNumbers(num1: 1, num2: 2)

3> 无返回值的函数

func greet(person: String) {
    print("hello, \(person)")
}
greet(person: "Lily")

4> 多返回值的函数

  • 为了让函数返回多个值作为一个复合的返回值,你可以使用元组类型作为返回类型,返回类型也可能为 nil,因此也可以是可选元组返回类型
func minMax(array: [Int]) -> (min: Int, max: Int)? {
    if array.isEmpty {
        return nil
    }
    var minValue = array[0]
    var maxValue = array[0]
    for value in array {
        if value < minValue {
            minValue = value
        } else if value > maxValue {
            maxValue = value
        }
    }
    return (minValue, maxValue)
}
if let array = minMax(array: [8, 101, -2, 10, 59]) {
    print("min is \(array.min), max is \(array.max)")
}

5> 函数实际参数标签和形式参数名

  • 每一个函数的形式参数都包含实际参数标签和形式参名。实际参数标签用在调用函数的时候;在调用函数的时候每一个实际参数前边都要写实际参数标签。形式参数名用在函数的实现当中。默认情况下,形式参数使用它们的形式参数名作为实际参数标签
  • 在提供形式参数名之前写实际参数标签,用空格分隔
  • 如果你为一个形式参数提供了实际参数标签,那么这个实际参数就必须在调用函数的时候使用标签
  • 如果对于函数的形式参数不想使用实际参数标签的话,可以利用下划线( _ )来为 这个形式参数代替显式的实际参数标签
//eat 为实际参数标签, food 为形式参数名
func eat(person: String, eat food: String) {
    print("\(person) started eating an \(food)")
}
eat(person: "zhangsan", eat: "apple")
//省略实际参数标签
func someFunction(_ firstNum: Int, secondNum: Int) {
    print("firstNum is \(firstNum),secondNum is \(secondNum)")
}
someFunction(1, secondNum: 2)

6> 默认形式参数值

  • 你可以通过在形式参数类型后给形式参数赋一个值来给函数的任意形式参数定义一个默认值
  • 如果定义了默认值,你就可以在调用函数时候省略这个形式参数
//定义函数的时候给一个默认的值
func someFunction(defaultNum: Int = 10) {
    print("num is \(defaultNum)")
}
someFunction(defaultNum: 6)   //输出: 6
someFunction()   //输出: 10

7> 可变形式参数

  • 一个可变形式参数可以接受零或者多个特定类型的值。当调用函数的时候你可以利用可变形式参数来声明形式参数可以被传入值的数量是可变的。可以通过在形式参数的类型名称后边插入三个点符号( …)来书写可变形式参数
  • 传入到可变参数中的值在函数的主体中被当作是对应类型的数组
func sum(numbers: Int...) -> Int {
    var sum = 0
    for num in numbers {
        sum += num
    }
    return sum
}
sum(numbers: 1, 2, 3, 4, 5)

8> 输入输出形式参数

  • 可变形式参数只能在函数的内部做改变。如果你想函数能够修改一个形式参数的值,而且你想这些改变在函数结束之后依然生效,那么就需要将形式参数定义为输入输出形式参数
  • 在形式参数定义开始的时候在前边添加一个 inout 关键字可以定义一个输入输出形式参数。输入输出形式参数有一个能输入给函数的值,函数能对其进行修改,还能输出到函数外边替换原来的值
  • 你只能把变量作为输入输出形式参数的实际参数,在将变量作为实际参数传递给输入输出形式参数的时候, 直接在它前边添加一个和符号 ( &) 来明确可以被函数修改
  • 输入输出形式参数不能有默认值,可变形式参数不能标记为 inout
//交换两个数
func swapTwoNumber(_ a: inout Int, _ b: inout Int = 20) {
    let temporaryA = a
    a = b
    b = temporaryA
}
var num1 = 8
var num2 = 10
swap(&num1, &num2)
print("num1 is \(num1), num2 is \(num2)")

3、函数类型和内嵌函数

1> 函数类型

  • 每一个函数都有一个特定的函数类型,它由形式参数类型,返回类型组成
    2> 使用函数类型
  • 你可以像使用 Swift 中的其他类型一样使用函数类型。例如,你可以给一个常量或变 量定义一个函数类型,并且为变量指定一个相应的函数
func addTwoNumber(num1: Int, num2: Int) -> Int {
    return num1 + num2
}
var mathFunction: (Int, Int) -> Int = addTwoNumber(num1:num2:)
print(mathFunction(2, 3))

//添加实际参数标签和之前没有实际参数标签的使用是一样的
func addTwoNumber(first num1: Int, second num2: Int) -> Int {
    return num1 + num2
}
let mathFunction2: (Int, Int) -> Int = addTwoNumber(first:second:)
print(mathFunction(1, 2))

3> 函数类型作为形式参数类型

  • 你可以利用使用一个函数的类型例如 (Int, Int) -> Int 作为其他函数的形式参数类型。 这允许你预留函数的部分实现从而让函数的调用者在调用函数的时候提供
func addTwoNumber(num1: Int, num2: Int) -> Int {
    return num1 + num2
}
//函数类型作为形式参数类型
func result(_ mathFunction: (Int, Int) -> Int, _ a: Int, _ b: Int) {
    print("result: \(mathFunction(a, b))")
}
result(addTwoNumber(num1:num2:), 2, 3)

4> 函数类型作为返回类型

  • 你可以利用函数的类型作为另一个函数的返回类型。写法是在函数的返回箭头( ->) 后立即写一个完整的函数类型
//前进
func stepForward(_ input: Int) -> Int {
    return input + 1
}
//后退
func stepBackWard(_ input: Int) -> Int {
    return input - 1
}
func chooseStepFunction (backward: Bool) -> (Int) -> Int {
    return backward ? stepBackWard : stepForward
}
var currentValue = -4
let moveNearerToZera = chooseStepFunction(backward: currentValue > 0)
print(moveNearerToZera)
while currentValue != 0 {
    print("\(currentValue)...")
    currentValue = moveNearerToZera(currentValue)
}
print("value is \(currentValue)")

5> 内嵌函数

  • 可以在函数的内部定义另外一个函数。这就是内嵌函数
  • 内嵌函数在默认情况下在外部是被隐藏起来的,但却仍然可以通过包裹它们的函数来调用它们。包裹的函数也可以返回它内部的一个内嵌函数来在另外的范围里使用
func chooseStepFunction(backward: Bool) -> (Int) -> Int {
    //前进
    func stepForward(_ input: Int) -> Int {
        return input + 1
    }
    func stepBackward(_ input: Int) -> Int {
        return input - 1
    }
    return backward ? stepBackward : stepForward
}
var currentValue = -4
let moveNearerToZera = chooseStepFunction(backward: currentValue > 0)
print(moveNearerToZera)
while currentValue != 0 {
    print("\(currentValue)...")
    currentValue = moveNearerToZera(currentValue)
}
print("value is \(currentValue)")