Swift 공식 문서의 여섯번 째 단원인 Functions을 읽고 정리를 해보려고 합니다.
Swift Apple 공식 문서 6챕터 Functions
Function은 해석하면 함수라는 뜻으로 특정 작업을 수행하는 독립적인 코드이다.
함수는 모두 이름을 가지고 있고 호출할 때 이름을 가지고 호출한다.
Swift에서는 매개 변수 이름이 없는 C언어 스타일부터 각 매개 변수에 대한 이름, 인수 레이블이 있는 복잡한 Objective-C 스타일의 함수까지 표현 가능하다.
매개 변수는 함수 호출을 간단하게 하기 위해 default 값을 줄 수도 있고 실행 후 전달된 매개 변수를 수정 가능하게 하는 입력 매개 변수로 전달될 수도 있다.
Swift의 모든 함수는 매개 변수의 타입과 반환 타입으로 구성된다.
이를 사용하면 다른 함수로 매개 변수를 전달하고 함수에서 함수로 쉽게 값들을 반환할 수 있다.
함수가 중첩되어 있으면 유용한 기능을 캡슐화하기 위해 다른 함수 속에 함수를 작성할 수도 있다.
함수를 정의할 때 함수에 매개 변수(parameter)라고 불리는 입력값의 타입을 선언할 수 있다.
물론 반환값(return type)이라고 하는 출력값의 타입도 선언할 수 있다.
매개 변수와 반환값은 선택사항으로 함수 정의에 반드시 필요한 것은 아니다.
모든 함수는 이름을 가진다. 함수를 사용하려면 이름을 가지고 호출하고 매개 변수를 타입에 맞게 입력해 주면 된다.
이때 매개 변수의 인수는 항상 선언될 때 순서와 동일하게 작성해야 한다.
func greet(person: String) -> String {
let greeting = "Hello, " + person + "!"
return greeting
}
print(greet(person: "Anna"))
// Prints "Hello, Anna!"
print(greet(person: "Brian"))
// Prints "Hello, Brian!"
func sayHelloWorld() -> String {
return "hello, world"
}
print(sayHelloWorld())
// Prints "hello, world"
func greet(person: String, alreadyGreeted: Bool) -> String {
if alreadyGreeted {
return greetAgain(person: person)
} else {
return greet(person: person)
}
}
print(greet(person: "Tim", alreadyGreeted: true))
// Prints "Hello again, Tim!"
greet(person:)
함수와 지금 선언한 greet(person:alreadyGreeted:)
함수는 다른 함수라는 것이다. func greet(person: String) {
print("Hello, \(person)!")
}
greet(person: "Dave")
// Prints "Hello, Dave!"
func printAndCount(string: String) -> Int {
print(string)
return string.count
}
func printWithoutCounting(string: String) {
let _ = printAndCount(string: string)
}
printAndCount(string: "hello, world")
// prints "hello, world" and returns a value of 12
printWithoutCounting(string: "hello, world")
// prints "hello, world" but does not return a value
위의 코드에서 printAndCount는 print 한 뒤 문자열이 가진 문자 수를 반환하는 함수이다.
printWithoutCounting 함수는 상수에 printAndCount의 반환값을 저장하는데 _ 를 사용해서 이를 사용하지는 않겠다는 표현을 해줬다.
이렇게 반환값을 사용하지 않겠다고 무시는 할 수 있지만 반환값이 있는 함수에게 반환하지 말라고는 할 수 없다. 값을 반환하지 않고 함수는 끝날 수 없다.
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)
}
(min: Int, max: Int)
타입을 반환한다고 정의한 것을 볼 수 있다. 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)
}
func greeting(for person: String) -> String {
"Hello, " + person + "!"
}
print(greeting(for: "Dave"))
// Prints "Hello, Dave!"
func anotherGreeting(for person: String) -> String {
return "Hello, " + person + "!"
}
print(anotherGreeting(for: "Dave"))
// Prints "Hello, Dave!"
func someFunction(firstParameterName: Int, secondParameterName: Int) {
// In the function body, firstParameterName and secondParameterName
// refer to the argument values for the first and second parameters.
}
someFunction(firstParameterName: 1, secondParameterName: 2)
func someFunction(argumentLabel parameterName: Int) {
// In the function body, parameterName refers to the argument value
// for that parameter.
}
func greet(person: String, from hometown: String) -> String {
return "Hello \(person)! Glad you could visit from \(hometown)."
}
print(greet(person: "Bill", from: "Cupertino"))
// Prints "Hello Bill! Glad you could visit from Cupertino."
func someFunction(_ firstParameterName: Int, secondParameterName: Int) {
// In the function body, firstParameterName and secondParameterName
// refer to the argument values for the first and second parameters.
}
someFunction(1, secondParameterName: 2)
func someFunction(parameterWithoutDefault: Int, parameterWithDefault: Int = 12) {
// If you omit the second argument when calling this function, then
// the value of parameterWithDefault is 12 inside the function body.
}
someFunction(parameterWithoutDefault: 3, parameterWithDefault: 6) // parameterWithDefault is 6
someFunction(parameterWithoutDefault: 4) // parameterWithDefault is 12
위의 코드에서 보듯 매개변수 선언 뒤에 = 12와 같이 값을 미리 정해줄 수 있는 것이다.
이렇게 하면 실제로 매개변수에 아무런 값이 입력되지 않으면 미리 정해둔 default value 값으로 함수를 실행할 수 있다.
이렇게 default Value를 사용할 때 매개 변수의 순서를 지켜주는 것이 좋다.
앞쪽에 default Value를 정의하지 않은 매개변수들을 두고 뒤쪽에 default value를 사용한 매개변수들을 두는 것이 좋다.
Variadic 매개변수는 매개변수의 입력값에 0개부터 아주 많게도 넣을 수 있는 매개변수를 말한다.
매개변수의 타입 뒤에 ...를 붙여 Variadic 매개변수를 사용할 수 있다
이렇게 입력된 variadic 매개변수는 적절한 타입의 Array로 함수 내에서 사용된다.
func arithmeticMean(_ numbers: Double...) -> Double {
var total: Double = 0
for number in numbers {
total += number
}
return total / Double(numbers.count)
}
arithmeticMean(1, 2, 3, 4, 5)
// returns 3.0, which is the arithmetic mean of these five numbers
arithmeticMean(3, 8.25, 18.75)
// returns 10.0, which is the arithmetic mean of these three numbers
함수의 매개변수는 기본적으로 상수이다.
함수 내에서 매개변수의 값을 변경하려고 하면 컴파일 오류를 발생한다.
하지만 매개변수를 수정하고 싶고 이를 함수가 끝난 후에도 유지하고 싶다면 in-out 매개변수를 사용하면 된다.
매개변수의 타입 앞에 inout 키워드를 써주면 in-out 매개변수를 만들 수 있다.
in-out 매개변수의 동작원리는 일단은 상수 타입으로 매개변수를 받고 이 매개변수를 수정 가능하게 하기 위해 함수에서 다시 전달돼서 기존의 매개변수 값을 대체하게 된다.
이런 방식으로 사용되다 보니 In-Out 매개변수에는 입력값으로 변수만 올 수 있다.
즉 상수가 올 수 없다는 것이다. in-out 매개변수를 호출해서 사용할 때는 변수의 이름 바로 앞에 &를 사용해 줘서 해당 변수가 함수 내에서 수정될 수 있다는 것을 알려줘야 한다.
이렇게 선언된 In-out 매개변수는 default 값과 variadic 매개변수를 사용할 수 없다.
func swapTwoInts(_ a: inout Int, _ b: inout Int) {
let temporaryA = a
a = b
b = temporaryA
}
var someInt = 3
var anotherInt = 107
swapTwoInts(&someInt, &anotherInt)
print("someInt is now \(someInt), and anotherInt is now \(anotherInt)")
// Prints "someInt is now 107, and anotherInt is now 3"
모든 함수는 함수 타입을 가지고 있다. 이는 매개변수의 타입과 반환값의 타입에 의해 결정된다.
func addTwoInts(_ a: Int, _ b: Int) -> Int {
return a + b
}
func multiplyTwoInts(_ a: Int, _ b: Int) -> Int {
return a * b
}
func printHelloWorld() {
print("hello, world")
}
var mathFunction: (Int, Int) -> Int = addTwoInts
print("Result: \(mathFunction(2, 3))")
// Prints "Result: 5"
mathFunction = multiplyTwoInts
print("Result: \(mathFunction(2, 3))")
// Prints "Result: 6"
let anotherMathFunction = addTwoInts
// anotherMathFunction is inferred to be of type (Int, Int) -> Int
func printMathResult(_ mathFunction: (Int, Int) -> Int, _ a: Int, _ b: Int) {
print("Result: \(mathFunction(a, b))")
}
printMathResult(addTwoInts, 3, 5)
// Prints "Result: 8"
위에서 선언한 addTwoInts를 재활용하여 위의 코드를 만들 수 있다.
위의 코드에 선언된 printMathResult 함수는 a, b 값을 입력받아 mathFunction이라는 함수 타입의 매개변수의 입력값으로 사용한 값을 출력하는 기능을 가지게 된다.
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 = 3
let moveNearerToZero = chooseStepFunction(backward: currentValue > 0)
// moveNearerToZero now refers to the stepBackward() function
print("Counting to zero:")
// Counting to zero:
while currentValue != 0 {
print("\(currentValue)... ")
currentValue = moveNearerToZero(currentValue)
}
print("zero!")
// 3...
// 2...
// 1...
// zero!
(Int) -> Int
함수 타입의 stepForward 함수와 stepBackward 함수를 선언하고 이를 chooseStepFunction의 반환값으로 사용했다. 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 moveNearerToZero = chooseStepFunction(backward: currentValue > 0)
// moveNearerToZero now refers to the nested stepForward() function
while currentValue != 0 {
print("\(currentValue)... ")
currentValue = moveNearerToZero(currentValue)
}
print("zero!")
// -4...
// -3...
// -2...
// -1...
// zero!