본문 바로가기

swift

[swift] 클로저 (closure)

클로저 (closure)

 

클로저는 실행가능한 코드 블럭을 말한다. (함수도 클로저의 일종)

이름을 정의할 필요가 없고, 매개변수와 리턴 값이 존재한다. (이름이 있는 클로저가 함수!)

일급 객체이다. ➡️ 변수, 상수로 저장, 전달인자로 전달 가능

{ (매개변수 목록) -> 반환타입 in
    실행 코드
}

// 매개 변수가 필요하지 않다면? 
{ () -> 반환타입 in
	실행 코드
}
// 함수를 사용할 경우
func sumFunction(a: Int, b: Int) -> Int {
	return a + b
}
var sumResult: Int = sumFunction(a: 1, b: 2)
print(sumResult)

// 클로저를 사용할 경우
var sum: (Int, Int) -> Int = { (a: Int, b: Int) -> Int) in
	return a + b
}
sumResult = sum(1, 2)
print(sumResult)

 

함수는 클로저의 일종이므로, sum 변수에는 함수도 할당할 수 있다.

sum = sumFunction(a: b:)
sumResult = sum(1, 2)

 

클로저는 주로 함수의 전달인자로 사용된다.

함수 내부에서 원하는 코드블럭을 실행할 수 있다.

let add: (Int, Int) -> Int
add = { (a: Int, b: Int) in
    return a + b
}

let substract: (Int, Int) -> Int
substract = { (a: Int, b: Int) in
    return a - b
}

let divide: (Int, Int) -> Int
divide = { (a: Int, b: Int) in
    return a / b
}

// method라는 이름으로 클로저를 넘겨줌 ! 
func calculate(a: Int, b: Int, method: (Int, Int) -> Int) -> Int {
    return method(a, b)
}

var calculated: Int

calculated = calculate(a: 50, b: 10, method: add)
print(calculated) // 60

calculated = calculate(a: 50, b: 10, method: substract)
print(calculated) // 40

calculated = calculate(a: 50, b: 10, method: divide)
print(calculated) // 5


//따로 클로저를 상수,변수에 넣어 전달하지 않고, 
//함수를 호출할 때 클로저를 작성하여 전달할 수도 있습니다.
calculated = calculate(a: 50, b: 10, method: { (left: Int, right: Int) -> Int in
    return left * right
})

print(calculated) // 500

후행 클로저

함수의 마지막 매개변수로 전달되는 클로저는 이 클로저는 함수 밖으로 빼 줄 수 있다.

// 기본 클로저 표현
func calculate(a: Int, b: Int, method: (Int, Int) -> Int) -> Int {
    return method(a, b)
}

// 후행 클로저로 변환
result = calculate(a: 10, b: 10) { (left: Int, right: Int) -> Int in
    return left + right
}

print(result)

 

반환 타입 생략

calculate(a: b: method:) 함수의 method 매개변수는 Int 타입을 반환할 것이라는 사실을 컴파일러도 알고 있다.

➡️ 클로저에서 리턴 타입이 Int라고 굳이 명시해주지 않아도 됨

대신 in 키워드는 생략할 수 없다.

result = calculate(a: 10, b: 10, method: { (left: Int, right: Int) in
    return left + right
})

print(result) 

// 후행클로저와 함께 사용
result = calculate(a: 10, b: 10) { (left: Int, right: Int) in
    return left + right
}

print(result) 

 

단축 인자이름

클로저의 매개변수 이름이 굳이 필요하지 않다면 단축 인자이름을 활용할 수 있다.

단축 인자이름은 클로저의 매개변수의 순서대로 $0, $1, ... 로 표현한다.

(매개변수의 타입도 컴파일러가 알고 있기 때문에 사용 가능한 것)

result = calculate(a: 10, b: 10, method: {
    return $0 + $1
})

print(result) 


// 후행 클로저와 함께 사용
result = calculate(a: 10, b: 10) {
    return $0 + $1
}

print(result)

 

암시적 반환 표현

클로저가 반환하는 값이 있다면 클로저의 마지막 줄의 결과값은 암시적으로 반환값으로 취급한다.

result = calculate(a: 10, b: 10) {
    $0 + $1
}

print(result)

// 간결하게 한 줄로 표현 가능
result = calculate(a: 10, b: 10) { $0 + $1 }

print(result) 

 

축약 전, 후 비교

//축약 전
result = calculate(a: 10, b: 10, method: { (left: Int, right: Int) -> Int in
    return left + right
})

//축약 후
result = calculate(a: 10, b: 10) { $0 + $1 }

너무 축약된 코드는 타인이 보거나, 시간이 지난 뒤에 볼 때 명시성이 떨어질 수 있으므로 적당히 축약해서 사용할 것!

'swift' 카테고리의 다른 글

[swift] sort와 sorted  (0) 2020.07.23
[swift] String.Index와 String.SubSequence(Substring)  (0) 2020.07.21
[swift] 값 타입 vs 참조 타입  (0) 2020.06.22
[swift] 열거형 (enum)  (0) 2020.06.22
[swift] 클래스(class)  (0) 2020.06.22