[Kotlin] 스코프 함수
스코프 함수(Scope Function)는 코드를 축약해서 표현할 수 있도록 도와주는 함수이다.
영역 함수라도고 한다.
사용법은 함수처럼 쓰지 않고 run, let처럼 괄호 없이 일종의 키워드 같이 사용할 수 있다.
lateinit과 함께 Safe Call 남용을 막아주는 역할도 하기 때문에 많이 사용하는 요소이다.
스코프 함수에는 run, let, apply, also, with이 있다.
1) run과 let으로 보는 스코프 함수
- run과 let은 자신의 함수 스코프(코드 블록) 안에서 호출한 대상을 this와 it로 대체해서 사용할 수 있다.
run
스코프 함수 안에서 호출한 대상을 this로 사용할 수 있다.
클래스 내부의 함수를 사용하는 것과 동일한 효과이기 때문에 this는 생략하고 메서드나 프로퍼티를 바로 사용할 수 있다.
다음 예제에서는 MutableList를 run 함수를 사용해서 스코프를 지정한 후 내부에서 size 프로퍼티를 직접 호출하였다.
var list = mutableListOf("Scope", "Function")
list.run {
val listSize = size
println("list의 길이 run = $listSize")
}
let
함수 영역 안에서 호출한 대상을 it으로 사용할 수 있다.
it을 생략할 수는 없지만 target 등 다른 이름으로 바꿀 수 있다.
다음 예제에서는 MutableList의 size 속성을 let 스코프 안에서 it.size로 호출하였다.
var list = mutableListOf("Scope", "Function")
list.let { // it -> 생략된 형태. it -> 대신에 target -> 등으로 변경 가능합니다.
val listSize = it.size // 모든 속성과 함수를 it.멤버로 사용할 수 있습니다.
println("리스트의 길이 let = $listSize")
}
2) this와 it으로 구분하기
- 위에서 봤듯이 스코프 함수는 자신을 호출한 대상을 this 또는 it으로 대체해서 사용할 수 있는데, 나머지 스코프 함수의 사용법을 두 가지로 구분해서 알아보겠다.
this로 사용되는 스코프 함수: run, apply, with
- 다음은 apply와 with의 사용 예제이다. 스코프 함수 안에서 this로 사용되기 때문에 메서드나 프로퍼티를 직접 호출한다.
var list = mutableListOf("Scope", "Function")
list.apply {
val listSize = size
println("리스트의 길이 apply = $listSize")
}
with (list){
val listSize = size
println("리스트의 길이 with = $listSize")
}
여기서 잠깐!
호출 대상이 null일 경우
with는 스코프 함수이긴 하지만 위의 2개와는 다르게 확장(Extension) 함수가 아니기 때문에 일반 함수처럼 사용된다.
따라서 호출하는 대상이 null일 경우에는 with보다는 apply나 run을 사용하는 것이 효율적이다.
target?.apply { /* 코드 */ }
it으로 사용되는 스코프 함수: let, also
- 다음은 let과 also를 사용하는 예제이다.
var list = mutableListOf("Scope", "Function")
list.let { target -> // it을 target 등과 같이 다른 이름으로 변경 가능합니다.
val listSize = target.size // target으로 변경했기에 멤버 접근은 target.속성 입니다.
println("리스트의 길이 let = $listSize")
}
list.also {
val listSize = it.size
println("리스트의 길이 also = $listSize")
}
3) 반환값으로 구분하기
위에서 사용해본 것만으로는 this와 it으로 사용되는 함수가 2개 이상씩 중복으로 있는 것처럼 보일 수 있다.
하지만 동일하게 this로 사용되는 함수라도 대입 연산자를 사용해서 값을 반환할 경우에는 용도가 달라진다.
결괏값을 반환할 경우, 스코프가 종료되는 시점에서의 반환값이 다르기 때문에 서로 다른 역할을 하는 스코프 함수가 필요하다.
호출 대상인 this 자체를 반환하는 스코프 함수: apply, also
apply를 사용하면 스코프 함수 안에서 코드가 모두 완료된 후 자기 자신을 되돌려 준다.
다음 예제에서 apply 스코프의 마지막 줄에서 count()를 호출했지만 마지막 코드와 상관없이 그냥 MutalbeList 자신을 돌려주기 때문에 Scope, Function에 Apply가 추가된 값이 출력된다.
also 또한 동일하게 동작한다.
var list = mutableListOf("Scope", "Function")
val afterApply = list.apply {
add("Apply")
count()
}
println("반환값 apply = $afterApply") // 반환값 apply = [Scope, Function, Apply]
val afterAlso = list.also {
it.add("Also")
it.count()
}
println("반환값 also = $afterAlso") // 반환값 also = [Scope, Function, Apply, Also]
마지막 실행 코드를 반환하는 스코프 함수: let, run, with
let, run, with의 결괏값을 반환하는 경우에는 앞의 2개와는 완전히 다른 결과가 나올 수 있으므로 주의해야 한다.
자기 자신이 아닌 스코프의 마지막 코드를 반환하기 때문이다.
다음 예제의 let에서 스코프 마지막 코드가 it.count()로 종료되었다.
apply나 also라면 마지막 코드에 상관없이 Scope, Function, Run이 출력되지만 let은 마지막 코드가 반환되기 때문에 출력값으로 리스트의 개수인 3이 출력된다.
run과 with도 마지막 코드가 반환된다.
var list = mutableListOf("Scope", "Function")
val lastCount = list.let {
it.add("Run")
it.count()
}
println("반환값 let = $lastCount") // 반환값 let = 3
val lastItem = list.run {
add("Run")
get(size-1)
}
println("반환값 run = $lastItem") // 반환값 run = Run
val lastItemWith = with(list){
add("With")
get(size-1)
}
println("반환값 with = $lastItemWith") // 반환값 with = With