본문 바로가기

IOS

Swift - 클로저 {} by yagom

728x90
반응형

코드의 블럭

일급 시민(first-citizen), (그럼 이급 시민은 뭐야?)

변수, 상수 등으로 저장, 전달인자로 전달이 가능

함수 == 이름이 있는 클로저

이름이 없는 클로저를 주의 깊게 봐야함

정의

{(매개변수 목록) -> 반환타입 in

  실행 코드

}

다음과 같이 매개변수 목록과 반환타입이 없다면 생략할 수 있다.

{() -> in

  실행 코드

}

 

// 함수를 사용한다면
func sumFuntion(a: Int, b: Int) -> {
	return a + b
}

// 함수를 사용한다면
var sumResult: Int = sumFunction(a: 1, b: 2)

print(sumResult) // 3

// 클로저의 사용
var sum: (Int, Int) -> Int = {(a:Int, b:Int) -> Int in
	return a + b
}

sumResult = sum(1, 2)
print(sumResult) // 3

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

sum = sumFunction(a:b:)

sumResult = sum(1, 2)
print(sumResult) // 3

 

 

함수의 전달인자로서의 클로저

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

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

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

// method라는 이름으로 클로저를 넘겨줄 것이야~
// calculate의 파라미터는 총 3개
// a, b, method
// a의 입력은 Int, b의 입력은 Int, method의 입력은 (Int, Int)
func calculate(a: Int, b: Int, method(Int, Int) -> Int){
	return method(a,b)
}

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: {(left: Int, right:Int) -> Int in
	return left * right
}

print(calculated) // 500

후행클로저

반환타입 생략, 단축 인자이름, 암시적 반환 표현

func calculate(a: Int, b: Int, method(Int, Int) -> Int) -> Int {
	return (a, b)
}

var reuslt: Int

// 클로저가 함수의 마지막 전달인자라면 마지막 매개변수 이름을 생략한 후
// 함수 소활호 외부에 클로저를 구현할 수 있다.

result = calculate(a: 10, b: 10){(left:Int, right:Int) -> Int in
	return left + right
} // 마지막 전달인자로 전달될 클로저, 즉 method를 클로저로 표현한것

print(result) // 20

반환타입 생략

calculate 함수의 method 매개변수는 Int 타입을 반환할 것이라는 사실을 컴파일러도 알기 때문에 굳이 클로저에서 반환타입을 명시해 주지 안하도 된다. 대신 in 키워든는 생략할 수 없다.

result = calculate(a: 10, b: 10, method: { (left: Int, right: Int) in
	return left + right
} // 반환타입 -> Int 생략

print(result) // 20


// 후행클로저와 함께 사용할 수도 있다.
reuslt = calculate(a: 10, b: 20){(left: Int, right: Int) in
	return left + right
}

// 메서드의 전달인자는 이미 앞에서 정의되어 있으므로 당연히 클로저는 앞의 타입을 가지고 있으므로 반환타입을 알고있다.

단축 인자 이름($0, $1 ...)

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

// 매개변수의 타입도 이전의 메소드에서 작성되어있기 때문에 컴파일러가 알고 있다.

result = calculate(a: 10, b: 10, method: {
	return $0 + $1
}) // 매개변수의 타입을 생략한다 $0은 첫번째 매개변수, $1은 두번째 매개변수 ... 그리고 in도 생략된다.

print(result) // 20

// 당연히 후행 클로저와 함께 사용할 수 있다.

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

print(result) // 20

암시적 반환 표현

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

result = calculate(a: 10, b: 10){
	$0 + $1
} // return 생략가능, 컴파일러가 반환값으로 인식함

print(result)

축약 클로저와 비축약 클로저의 비교

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

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

print(result) // 20
반응형