import UIKit

// 1. 후행 클로저: 함수의 매개변수 마지막으로 전달되는 클로저는 후행클로저(trailing closure)로 함수 밖에 구현 가능
// 2. 반환타입 생략: 컴파일러가 클로저의 타입을 유추할 수 있는 경우 매개변수, 반환 타입을 생략 가능
// 3. 단축 인자 이름: 전달인자의 이름이 굳이 필요없고, 컴파일러가 타입을 유추할 수 있는 경우 축약된 전달인자 이름($0, $1, $2 ...)을 사용 가능
// 4. 암시적 반환 표현: 반환 값이 있는 경우, 암시적으로 클로저의 맨 마지막 줄은 return 키워드를 생략하더라도 반환 값으로 취급


/// 기본 클로저 표현
// 클로저를 매개변수로 갖는 함수 calcaulated(a: b: method:)와 결과값을 저장할 변수 result 선언
func calculate(a: Int, b: Int, method: (Int, Int) -> Int) -> Int {
    return method(a, b)
}

var result: Int

/// 1. 후행 클로저
// 클로저가 함수의 마지막 전달인자일때, 마지막 매개변수 이름을 생략한 후 함수 소괄호 외부에 클로저를 구현 가능
result = calculate(a: 10, b: 10) { (left: Int, right: Int) -> Int in
    return left + right
}

print(result) // 20


/// 2. 반환타입 생략
// calculate(a:b:method:) 함수의 method 매개변수는 Int 타입을 반환할 것이라는 사실을 컴파일러도 알기 때문에 굳이 클로저에서 반환타입을 명시해 주지 않아도 된다. 대신 in 키워드는 생략 불가
result = calculate(a: 10, b: 10, method: { (left: Int, right: Int) in
    return left + right
})

print(result) // 20


/// 3. 단축 인자이름
// 클로저의 매개변수 이름이 굳이 불필요하다면 단축 인자이름을 활용 가능.
// 단축 인자이름은 클로저의 매개변수의 순서대로 $0, $1, $2 ... 처럼 표현
result = calculate(a: 10, b: 10, method: {
    return $0 + $1
})

print(result) // 20

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

print(result) // 20


/// 4. 암시적 반환 표현
// 클로저가 반환하는 값이 있다면 클로저의 마지막 줄의 결과값은 암시적으로 반환값으로 취급
result = calculate(a: 10, b: 10){
    $0 + $1
}

print(result) // 20

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

print(result) // 20


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

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

print(result) // 20

// 너무 축약된 코드는 타인이 보거나, 시간이 지난 뒤에 볼 때 명시성이 떨어질 수 있으므로 적정선에서 축약

'iOS > 문법' 카테고리의 다른 글

클로저 기본  (0) 2021.08.19
변수 상수  (0) 2021.08.09
import UIKit

/// 1. 클로저
// 실행가능한 코드 블럭
// 함수와 다르게 이름정의는 필요하지 않지만, 매개변수 전달과 반환 값이 존재
// 함수는 이름이 있는 클로저
// 일급객체로 전달인자, 변수, 상수 등에 저장 및 전달 가능


/// 2. 기본 클로저 문법
// 클로저는 중괄호 {}로 감싼다
// 괄호를 이용해 파라미터를 정의
// -> 을 이용해 반환 타입을 명시
// "in" 키워드를 이용해 실행 코드와 분리
/*
 {(매개변수 목록) -> 반환타입 in
    실행 코드
 }
 */


/// 3. 클로저 사용
// sum이라는 상수에 클로저를 할당
let sum: (Int, Int) -> Int /* 클로저 타입 표현 */ = {(a: Int, b: Int) in
    return a + b
}

let sumResult: Int = sum(1, 2)
print(sumResult) // 3


/// 4. 함수의 전달인자로서의 클로저
// 클로저는 주로 함수의 전달인자로 많이 사용
// 함수 내부에서 원하는 코드블럭을 실행
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
}

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

'iOS > 문법' 카테고리의 다른 글

클로저 고급 - 다양한 클로저 표현  (0) 2021.08.19
변수 상수  (0) 2021.08.09

'iOS > 문법' 카테고리의 다른 글

클로저 고급 - 다양한 클로저 표현  (0) 2021.08.19
클로저 기본  (0) 2021.08.19

+ Recent posts