- 함수 타입
- 고차 함수와 코드를 구조화할 때 고차 함수를 사용하는 방법
- 인라인 함수
- 비로컬 return과 레이블
- 무명 함수
람다를 인자로 받거나 반환하는 함수인 **고차 함수(high order function)**를 만드는 방법을 다루어 보자.
고차 함수로 코드를 더 간결하게 다듬고 코드 중복을 없애고 더 나은 추상화를 구축하는 방법을 살펴본다.
또한 람다를 사용함에 따라 발생할 수 있는 성능상 부가 비용을 없애고 람다 안에서 더 유연하게 흐름을 제어할 수 있는 코틀린 특성인 인라인(inline) 함수에 대해 배워보자.
- 고차 함수는 다른 함수를 인자로 받거나 함수를 반환하는 함수다. 코틀린에서는 람다나 함수 참조를 사용해 함수를 값으로 표현할 수 있다. 따라서, 고차 함수는 람다나 함수 참조를 인자로 넘길 수 있거나 람다나 함수 참조를 반환하는 함수다.
- e.g.) 표준 라이브러리 함수
filter
는 술어 함수를 인자로 받으므로 고차 함수다.-
list.filter { x > 0 }
-
// 타입 추론
val sum = {x: Int, y: Int -> x + y}
val action = { println(42) }
// 함수 타입 명시
val sum: (Int, Int) -> Int = {x, y -> x + y}
val action: () -> Unit = { println(42) }
->
좌측: 함수의 파라미터을 괄호 안에 명시->
우측: 함수의 반환 타입을 명시- 함수 타입을 선언할 때는 반환 타입을 반드시 명시해야 한다. (Unit도 마찬가지!)
- 함수 타입을 지정할 경우 람다의 파라미터 타입을 유추할 수 있으므로 람다 내에서 파라미터 타입을 생략할 수 있다.
// 반환 타입이 널이 될 수 있는 타입
var canReturnNull: (Int, Int) -> Int? = { x,y -> null }
// 함수 타입 변수 자체가 널이 될 수 있는 타입
var funOrNull: ((Int, Int) -> Int)? = null
- 함수 타입에서도 반환 타입이 널이 될 수 있는 타입으로 지정할 수 있다.
- 함수 타입 변수 자체를 널이 될 수 있는 타입으로도 사용이 가능하다.
- 코드 상으로는 단순히 괄호의 유무의 차이이지만 반환 타입이 널이 될 수 있는 타입 과 함수 타입 변수 자체가 널이 될 수 있는 타입은 큰 차이가 있다는 것을 유의해야 한다.
파라미터 이름과 함수 타입
fun performRequest (url: String, callback: (code: Int, conetnt: String) -> Unit) { /* TODO */ } val url = "http://kotl.in" performRequest(url) { code, content -> /* TODO */ } // API에서 제공하는 이름을 람다에서 사용 performRequest(url) { code, page -> /* TODO */ } // 원하는 이름으로 붙여서 사용
- 파라미터 이름은 타입 검사시 무시된다.
- 람다에서 호출하는 파라미터 이름이 꼭 함수 타입 선언의 파라미터 이름과 일치하지 않아도 된다.
- 함수 타입에 인자 이름을 추가하는 것이 가독성과 IDE를 사용함에 있어서 더 좋기 때문에 파라미터 이름을 사용하는 것을 권장한다.
fun twoAndThree(operation: (Int, Int) -> Int) {
val result = operation(2, 3)
println("The result is $result")
}
twoAndThree { a, b -> a + b}
// 결과: The result is 5
twoAndThree() { a, b -> a * b }
// 결과: The result is 6
- 인자로 받은 함수를 호출하는 구문은 일반 함수를 호출하는 구문과 같다.
String에 대한 filter 구현 해보기
fun String.filter(predicate: (Char) -> Boolean): String { val sb = StringBuilder() for (index in 0 until length) { val element = get(index) if (predicate(element)) sb.append(element) } return sb.toString() } println("ab1c".filter {it in 'a'..'z'}) // 결과: abc
String
: 수신 객체 타입predicate
: 파라미터 이름(Char) -> Boolean
: 파라미터 함수 타입(Char)
: 파라미터로 받는 함수의 파라미터 타입Boolean
: 파라미터로 받는 함수의 반환 타입
컨파일된 코드 안에서 함수 타입은 일반 인터페이스로 바뀐다.
함수 타입의 변수는 FunctionN 인터페이스를 구현하는 객체를 저장한다.
- 코틀린 표준 라이브러리는 함수 인자의 개수에 따라 Function0 (인자가 없는 함수), Function1<P1, R>(인자가 하나인 함수) 등의 인터페이스를 제공한다.
- 각 인터페이스에는
invoke()
메소드 정의가 하나 들어있으며invoke()
를 호출하면 함수를 실행할 수 있다.- 함수 타입인 변수는 FunctionN 인터페이스를 구현하는 클래스의 인스턴스를 저장
- 그 클래스의
invoke()
메소드 본문에는 람다의 본문을 저장
말로 보면 어려우니 코드로 자바에서 코틀린의 함수 타입을 사용해보자. (자바8 에서!)
// 코틀린에서 선언
fun processTheAnswer(f: (Int) -> Int) {
println(f(42))
}
// 자바 8 에서 위 코틀린 코드 호출
processTheAnswer(number -> number + 1);
// 결과: 43
//자바 8 이전에서 위 코틀린 코드 호출
processTheAnswer(
new Function1<Integer, Integer>() {
@Override
public Integer invoke(Integer number) {
System.out.println(number);
return number + 1;
}
});
// 결과: 43
- 자바 8 이후의 버전은 람다를 그대로 넘기면 자동으로 함수 타입의 값으로 변환된다.
- 하지만 자바 8 이전의 버전인 경우 필요한 FunctionN 인터페이스의
invoke()
메소드를 구현하는 무명 클래스를 넘겨주어야 한다.
// 코틀린 표준 라이브러리르 이용하여 코틀린의 람다를 인자로 받는 확장 함수 호출
List<String> strings = new ArrayList<>();
strings.add("42");
CollectionsKt.forEach(strings, s -> { // strings는 확장 함수의 수신 객체
System.out.println(s)
return Unit.INSTANCE; // Unit 타입의 값을 명시적으로 반환해야 함
});
- 코틀린 Unit 타입에는 값이 존재하므로 자바에서는 그 값을 명시적으로 return 해주어야 한다.
- 즉,
(String) -> Unit
처럼 반환 타입이 Unit인 경우 void를 반환하는 자바 람다를 넘길 수 없다.
3장에서 살펴본 joinToString 함수를 함수 타입의 파라미터에 대한 디폴트 값을 지정하여 가공해보자.
fun <T> Collection<T>.joinToString(separator: String = ", ",
prefix: String = "",
postfix: String = "",
transform: (T) -> String = { it.toString() } // 함수 타입 파라미터를 선언하면서 람다를 디폴트 값으로 지정
): String {
val result = StringBuilder(prefix)
for ((index, element) in this.withIndex()) {
if (index > 0) result.append(separator)
result.append(transform(element))
}
result.append(postfix)
return result.toString()
}
val letters = listOf("Alpha", "Beta")
// 디폴트 람다 사용
println(letters.joinToString())
// 결과: Alpha, Beta
// 직접 람다 전달
println(letters.joinToString { it.toLowerCase() })
// 결과: alpha, beta
// 이름이 붙은 인자 구문을 사용해 여러 인자를 전달
println(letters.joinToString(separator = "! ", postfix = "! ", transform = { it.toUpperCase() }))
//결과: ALPHA! BETA!
위와 다른 방법으로 널이 될 수 있는 함수 타입을 사용할 수 있다.
다만, 널이 될 수 있는 함수 타입을 사용하는 경우 그 함수를 직접 호출할 수 없다. 또한, NPE 발생 가능성이 있으므로 컴파일을 거부하게 된다
그런 경우 null 여부를 명시적으로 검사하면 된다.
fun foo(callback: (() -> Unit)?) {
// TODO
if (callback != null){
callback()
}
}
- 전에 말했듯이 함수 타입은
invoke()
메소드를 구현하는 인터페이스다. - 이 사실을 기억하면 일반 메소드처럼
invoke()
도 안전한 호출 구문(callback?.invoke()
)으로 호출할 수 있다는 것을 알 수 있다.
fun <T> Collection<T>.joinToString(separator: String = ", ",
prefix: String = "",
postfix: String = "",
transform: ((T) -> String)? = null // 널이 될 수 있는 함수 타입의 선언
): String {
val result = StringBuilder(prefix)
for ((index, element) in this.withIndex()) {
if (index > 0) result.append(separator)
val str = transform?.invoke(element) ?: element.toString()
result.append(str)
}
result.append(postfix)
return result.toString()
}
enum class Delivery { STANDARD, EXPEDITED }
class Order(val itemCount: Int)
fun getShippingCostCalculator(delivery: Delivery): (Order) -> Double {
if (delivery == Delivery.EXPEDITED) {
return { order -> 6 + 2.1 * order.itemCount }
}
return { order -> 1.2 * order.itemCount}
}
val calculator = getShippingCostCalculator(Delivery.EXPEDITED)
println("Shipping costs ${calculator(Order(3))}")
// 결과: Shipping costs 12.3
- 다른 함수를 반환하는 함수를 정의하려면 함수의 반환 타입으로 함수 타입을 지정해야 한다.
- 타입 추론을 믿고 생략할 경우
error: type mismatch: inferred type is (???) -> ??? but Unit was expected
컴파일 에러를 맞이하게 된다.
- 타입 추론을 믿고 생략할 경우
data class Person(val firstName: String, val lastName: String, val phoneNumber: String?)
class ContactListFilters {
var prefix: String = ""
var onlyWithPhoneNumber: Boolean = false
fun getPredicate(): (Person) -> Boolean { // 함수를 반환하는 함수를 정의
val startsWithPrefix = { p: Person ->
p.firstName.startsWith(prefix) || p.lastName.startsWith(prefix)
}
if (!onlyWithPhoneNumber) {
return startsWithPrefix
}
return { startsWithPrefix(it) && it.phoneNumber != null } // 람다를 반환
}
}
val contacts = listOf(Person("Dmitry", "Jemerov", "123-4567"), Person("Svetlana", "Isakova", null))
val contactListFilters = ContactListFilters()
with (contactListFilters) {
prefix = "Dm"
onlyWithPhoneNumber = true
}
println(contacts.filter(contactListFilters.getPredicate()))
// 결과: [Person(firstName=Dmitry, lastName=Jemerov, phoneNumber=123-4567)]
getPredicate()
메소드는filter
함수에 넘길 수 있는 함수를 반환한다.- 이처럼 함수 타입을 사용하면 함수에서 함수를 쉽게 반환할 수 있다.
다시 기억하자!
with()
- with 함수는 파라미터가 2개인 메소드로 첫 번째 인자는 객체를 두 번째 인자는 람다를 받는다.
- 첫 번째 인자로 받은 객체를 두 번째 인자로 받은 람다의 수신 객체로 만든다.
태형 Think !
- 위의 with() 함수가 어떻게 동작하는 것인지 이해가 가지 않는다.. 이야기 해보자..!
data class SiteVisit(val path: String, val duration: Double, val os: OS)
enum class OS { WINDOWS, LINUX, MAC, IOS, ANDROID }
val log = listOf(SiteVisit("/", 34.0, OS.WINDOWS),
SiteVisit("/", 22.0, OS.MAC),
SiteVisit("/login", 12.0, OS.WINDOWS),
SiteVisit("/signup", 8.0, OS.IOS),
SiteVisit("/", 16.3, OS.ANDROID))
// 윈도우 사용자의 평균 방문 시간 가져오기
val averageWindowsDuration = log.filter { it.os == OS.WINDOWS }
.map(SiteVisit::duration)
.average()
println(averageWindowsDuration)
// 결과: 23.0
// OS별 사용자 평균 방문 시간 가져오는 확장 함수 정의
fun List<SiteVisit>.averageDurationFor(os: OS) =
filter { it.os }.map(SiteVisit::duration).average()
println(log.averageDurationFor(OS.WINDOWS))
// 결과: 23.0
println(log.averageDurationFor(OS.MAC))
// 결과: 22.0
// 하드코딩한 필터를 사용해 방문 데이터 분석하기
val averageMobileDuration = log.filter { it.os in setOf(OS.IOS, OS.ANDROID) }
.map(SiteVisit::duration)
.average()
println(averageMobileDuration)
// 결과: 12.15
// 고차 함수를 사용해 중복 제거하기
fun List<SiteVisit>.averageMobileDurationFor(predicate: (SiteVisit) -> Boolean) =
filter(predicate).map(SiteVisit::duration).average()
- 확장 함수를 정의하며 가독성이 좋아진 것을 볼 수 있다.
- 함수 타입은 코드 중복을 줄일 때 상당한 도움이 된다.
- 함수 타입을 통해 전략 패턴(Strategy Pattern)을 손쉽게 적용할 수 있다.
리마인드 할 내용
- 보통 람다를 무명 클래스로 컴파일하지만 그렇다고 람다 식을 사용할 때마다 새로운 클래스를 만들지는 않는다.
- 람다가 변수를 포획하면 람다가 생성되는 시점마다 새로운 무명 클래스 객체가 생긴다.
리마인드 할 내용에 따르면 무명 클래스 생성에 따른 부가 비용이 발생한다. 따라서 일반 함수를 사용한 구현보다 덜 효율적이다.
inline
변경자를 어떤 함수에 붙이면 컴파일러는 그 함수를 호출하는 모든 문장을 함수 본문에 해당하는 바이트코드로 바꿔치기 해주며 그로 인해 일반 명령문만큼 효율적인 코드를 생성하게 할 수 있다.
inline
: 함수를 호출하는 코드를 함수를 호출하는 바이트코드 대신에 함수 본문을 번역한 바이트 코드로 컴파일한다는 뜻
// 인라인 함수 정의하기
inline fun <T> synchronized(lock: Lock, action: () -> T): T {
lock.lock()
try {
return action()
} finally {
lock.unlock()
}
}
val l = Lock()
synchronized(l) {
// TODO
}
위 코드에 대한 의문
- 코틀린에서는
try catch
구문에서catch
를 생략해도 되는 것인가?
- 코틀린뿐만 아니라 자바에서도
try-finally
구문이 있는 것으로 보인다.- 참고 자료
fun foo(l: Lock) {
println("Before sync")
synchronized(l) {
println("Action")
}
println("After sync")
}
을 컴파일 하면?
fun __foo__(l: Lock) {
println("Before sync")
l.lock() // synchronized 함수가 인라이닝된 코드 시작
try {
println("Action") // 람다 코드의 본문이 인라이닝된 코드
} finally {
l.unlock()
} // synchronized 함수가 인라이닝된 코드 끝
println("After sync")
}
- synchronized 함수의 본문뿐 아니라 synchronized에 전달된 람다의 본문도 함께 인라이닝된다.
class LockOwnser(val lock: Lock) {
fun runUnderLock(body: () -> Unit) {
synchronized(lock, body)
}
}
- 위의 코드의 경우 인라인 함수를 호출하는 코드 위치에서는 변수에 저장된 람다의 코드를 알 수 없다.
- 따라서 람다 본문은 인라이닝되지 않고 synchronized 함수의 본문만 인라이닝된다.
위 코드를 컴파일 하면?
class LockOwner(val lock: Lock) {
fun __runUnderLock__(body: () -> Unit) {
lock.lock()
try {
body()
} finally {
lock.unlock()
}
}
}
정리 사항
- 한 인라인 함수를 두 곳에서 각각 다른 람다를 사용해 호출한다면 그 두 호출은 각각 따로 인라이닝된다.
- 인라인 함수의 본문 코드가 호출 지점에 복사되고 각 람다의 본문이 인라인 함수의 본문 코드에서 람다를 사용하는 위치에 복사된다.
함수 본문에서 파라미터로 받은 람다를 호출한다면 그 호출을 쉽게 람다 본문으로 바꿀 수 있다.
파라미터로 받은 람다를 다른 변수에 저장하고 나중에 그 변수를 사용하는 경우 람다를 인라이닝할 수 없다.
- 람다를 표현하는 객체(변수)가 어딘가는 존재해야 하기 때문!!
인라인 함수의 본문에서 람다 식을 바로 호출하거나 람다 식을 인자로 전달받아 바로 호출하는 경우에는 그 람다를 인라이닝할 수 있다.
단, 그 외에 경우 Illegal usage of inline-parameter
와 함께 컴파 에러가 발생한다.
inline fun <T> synchronized(lock: Lock, action: () -> T): T {
val someA = action // ‐> 컴파일 에러
}
// Illegal usage of inline-parameter 'action' ...
fun <T,R> Sequence<T>.map(transform: (T) -> R): Sequence<R> {
return TransformingSequence(this, transform)
}
// 파라미터 transform은 인라이닝하지 않은 함수(함수 인터페이스를 구현하는 무명 클래스 인스턴스)로 표현하여야 한다.
인라이닝 금지 방법: noinline
inline fun foo(inlined: () -> Unit, noinline notInlined: () -> Unit) {
// TODO
}
- 외부 모듈이나 서드파티 라이브러리에 있는 인라인 함수를 호출해 사용하거나 자바에서 코틀린 인라인 함수를 호출하는 경우 컴파일러는 인라인 함수를 인라이닝하지 않고 일반 함수 호출로 컴파일한다.
noinline
을 사용해야 하는 다른 상황은 추후에 더 살펴보도록 하자
코틀린의 filter나 map 등 컬렉션 함수도 인라인 함수다.
- 즉, 함수 인라이닝을 믿고 성능에 신경쓰지 않아도 된다.
data class Person(val name: String, val age: Int)
...
people.filter { it.age > 30} // 중간 리스트 생성
.map(Person::name)
- filter와 map을 연쇄할 경우 중간 리스트를 만든다.
- 시퀀스를 사용할 경우 중간 리스트로 인한 부가 비용을 줄일 수 있다.
- 중간 시퀀스는 람다를 필드에 저장하는 객체로 표현하고 최종 연산은 중간 시퀀스에 있는 여러 람다를 연쇄 호출
- 위 동작에 의해 람다를 저장하게 되므로 시퀀스는 람다를 인라인하지 않게 된다.
위의 이유로 모든 컬렉션에 asSequence
를 붙이면 안되며 크기가 작은 컬렉션의 경우 오히려 일반 컬렉션 연산이 성능이 더 좋을 수도 있으니 컬렉션 크기가 큰 경우에만 시퀀스를 통해 성능을 향상하자.
-
일반 함수 호출의 경우 JVM이 이미 강력하게 인라이닝을 지원한다.
- JVM은 코드 실행을 분석해 가장 이익이 되는 방향으로 호출을 인라이닝함
- 위와 같은 과정은 **JIT**에서 일어남
-
람다를 인자로 받는 함수를 인라이닝하면 이익이 많다.
- 람다 호출 시 발생하는 부가 비용 감소
- 람다를 표현하는 클래스와 람다 인스턴스에 해당하는 객체 생성 감소
- 현재 JVM은 함수 호출과 람다를 인라이닝 해주지 못함
태형 Think !
- JVM 단에서 함수 호출과 람다를 인라이닝 해주지 못하니 직접 인라인 함수를 명시해주면 인라이닝이 되기 때문인 것 같다.
넌로컬(non-local) 반환
과 같이 일반 람다에서는 사용할 수 없는 몇가지 기능을 사용할 수 있다.
-
inline 함수를 만들 때는 코드 크기에 주의가 필요하다.
- 바이트 코드를 증가시키므로 inline 함수는 크기가 작아야 한다.
일반적으로 자원 관리를 할 때에 try-finally
문을 사용하여 finally
블록에서 자원을 해제하는 방식을 사용한다.
코틀린 라이브러리의 withLock
val l: Lock = ...
l.withLock {
// 락에 의해 보호되는 자원을 사용
}
withLock
함수: 락을 잠근 다음에 주어진 동작을 수행하며 Lock 인터페이스의 확장함수이다.
자바7에 생긴 try-with-resource
static String readFirstLineFromFile(String path) throw IOException {
try(BufferedReader br = new BufferedReader(new FileReader(path))) {
return br.readLine()
}
}
- 리소스를 직접 해제해주지 않아도 리소스를
close()
해준다. - 단, AutoCloseable을 구현한 객체에 대해서만
close()
함수를 호출해준다. - try-with-resource 참고 자료
try-with-resource
를 대체하는 코틀린의 use
fun readFirstLineFromFile(path: String): String {
BufferedReader(FileReader(path)).use { br ->
return br.readLine() // 람다의 종료가 아닌 readFirstLineFromFile 함수를 끝내며 값 반환
}
}
- closeable 자원에 대한 확장 함수이며 인라인 함수이다.
넌로컬(non-local) 반환: 자신을 둘러싸고 있는 블록보다 더 바깥에 있는 다른 블록을 반환하게 만드는 return 문
data class Person(val name: String, val age: Int)
val people = listOf(Person("Alice", 29), Person("Bob", 31))
fun lookForAlice(people: List<Person>) {
people.forEach {
if (it.name == "Alice") {
println("Found !")
return
}
}
println("Alice is not found")
}
- 람다 내에서
return
을 사용할 경우 란다로부터만 반횐되는 것이 아니라 람다를 호출하는 함수가 실행을 끝내고 반환된다. - 인라인 함수에 전달된 람다의 경우에만 넌로컬 반환이 가능하다.
- 예제에서
forEach
는 인라인 함수이기 때문에 사용이 가능하다.
- 예제에서
- 인라인되지 않는 함수에 전달된 람다에서는 return을 사용할 수 없다.
로컬(local) 반환: 말 그대로 자신을 둘러싸고 있는 블록을 반환시키는 return 문
- 람다 내에서 사용될 경우 for 루프의 break와 비슷한 역할을 한다.
- 넌로컬 반환과 로컬 반환을 비교하기 위해 **레이블(label)**을 사용한다.
fun lookForAlice(people: List<Person>) {
people.forEach label@{
if (it.name == "Alice") return@label
}
println("Alice might be somewhere")
}
lookForAlice(people)
//결과: Alice might be somewhere
fun lookForAlice(people: List<Person>) {
people.forEach {
if (it.name == "Alice") return@forEach
}
println("Alice might be somewhere")
}
lookForAlice(people)
//결과: Alice might be somewhere
- 람다를 여는 중괄호({) 앞에서 람다 레이블 이름 뒤에 @ 문자를 추가하여 명시하고 return 키워드 뒤에 @ 문자와 레이블을 명시하면 된다.
- 인라인 함수의 이름을 return 뒤에 레이블로 사용해도 가능하다.
- 람다 식의 레이블을 명시할 경우 함수 이름을 레이블로 사용할 수 없다.
레이블이 붙은 this 식
- this 식의 레이블에도 동일한 규칙이 적용된다.
- 수신 객체 지정 람다 앞에 레이블을 붙인 경우 this 뒤에 그 레이블을 붙여서 묵시적인 컨텍스트 객체를 지정할 수 있다.
println(StringBuilder().apply sb@{ listOf(1, 2, 3).apply{ this@sb.append(this.toString()) } })
fun lookForAlice(people: List<Person>) {
people.forEach(fun (person) { // 람다식 대신 무명 함수를 사용
if (person.name == "Alice") return // 가장 가까운 함수에서 return 처리가 됨
println("${person.name} is not Alice")
})
}
people.filter(fun (person): Boolean {
return person.age < 30
})
// 식을 본문으로 하는 무명 함수의 반환 타입은 생략할 수 있음
people.filter(fun (person) = person.age < 30)
- Kotlin IN Action / 출판사: 에이콘