Swift의 Closure
- Swift의 Clousre를 다루기 전에 알아야하는 개념이 있음
- 일급 객체에 대해서 알아야 함
일급 객체란?
- 다음과 같은 조건을 만족하는 것을 일급 객체라고 말함
- 변수나 데이터 구조 안에 담을 수 있는가
- 파라미터로 전달할 수 있는가
- 반환값(return value)으로 사용할 수 있는가
- 할당에 사용된 이름과 관계없이 고유한 구별이 가능한가
- 동적으로 프로퍼티 할당이 가능한가
- 우리가 Swift에서 알고 있는 대부분은 일급 객체라고 할 수 있지만,
- Method는 일급 객체가 아니고, Closure가 존재하는 이유이기도 함
Closure의 유형
- Closure의 유형은 3가지로 나눠볼 수 있음
- Global function
- Nested function
- Closure expressions
Global function
- Global Function은 우리가 흔히 알고 있는 함수임
- 클래스 밖의 함수라고 할 수 있겠음
Nested function
- Nested function은 중첩 함수라고 말함
- 함수 내부에서 다시 함수를 정의해서 사용하는 함수
- 외부에는 숨겨져 있고, 선언된 함수 내부에서만 호출 가능
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
}
Closure expressions
- 이것이 흔히 클로저라고 불리는 유형임
- 문법은 다음과 같음
{ (parameters) -> ReturnType in
statements
}
클로져의 변수 할당
- 클로저는 일급 객체이기 때문에 변수에 할당될 수 있음
let closureValue = { (name:String) in print(name) }
closureValue ("woozoobro")
func closureOperation(then closure: () -> Void) {
}
클로저 축약하기
- Swift에서의 클로저는 다른 언어에서보다 유연함
- 그래서 보다 자유롭게 축약하거나 변형 가능
let string = "Hello, world!".transformWords(using: { word in
return word.lowercased()
})
let string = "Hello, world!".transformWords { $0.lowercased() }
- 후행 클로저를 사용해 함수의 파라미터를 축약할 수 있다.
- $0과 같은 인자 값을 사용해 첫 번째 인자값을 대체할 수 있다.
- 단일 표현 클로저에서는 return 키워드를 생략할 수 있다.
Trailing Closures
- Trailing Closure로 마지막 파라미터 값으로 들어오는 Closure를 생략할 수 있음
func someFunction (closure: () -> Void) {
}
someFunction(closure: {
})
someFunction() {
}
Capturing Values
- Closure는 주변의 value를 포착(Capture)함
- 다음 코드를 보자
func makeIncrementer(forIncrement amount: Int) -> () -> Int {
var runningTotal = 0
func incrementer() -> Int {
runningTotal += amount
return runningTotal
}
return incrementer
}
- incrementer 함수를 떼어놓고 보자.
- 없는 runningTotal과 amount를 사용하고 있음
func incrementer() -> Int {
runningTotal += amount
return runningTotal
}
- 그래서 주변 value를 capture 해서 사용함
- 그래서 다음과 같이 코드를 반복 실행하면, 값이 계속 증가하게 됨
let incrementByTen = makeIncrementer(forIncrement: 10)
incrementByTen()
incrementByTen()
incrementByTen()
Escaping Closure
- @escaping을 통해 escaping closure를 사용할 수 있음
- Escaping Closure는 기본값인 nonescape closure와 다른 점이 몇가지 있음
- 다음 코드와 같이 전달 받은 closure를 함수 외부에서 사용할 수 없음
var completionHandlers: [() -> Void] = []
func someFunctionWithEscapingClosure(completionHandler: @escaping () -> Void {
completionHandlers.append(completionHandler)
}
- 또, Escaping Closure는 함수가 종료된 뒤에도 메모리에 잡아둠
- 그래서 비동기 프로그래밍을 할 때 유리
https://bbiguduk.gitbook.io/swift/