Java에서와 마찬가지로,
Kotlin도 Genenric을 지원하는데요.
오늘은 Kotlin에서의 Generic에 대해 정리해 보겠습니다.
1. Class에서의 Generic
1-1. Generic
Generic은,
class 혹은 함수를 사용하는 시점에,
사용할 타입을 지정하도록 하는 것 입니다.
코드를 보면서 이해해 보겠습니다.
아래 코드를 보면,
클래스이름 다음에 <T>라는 타입을 붙인것이 보입니다.
Generic타입임을 의미하는 것 인데요.
Return값의 타입을 나타내는 곳과는 다른 곳으로,
위치가 조금 특이합니다.
아래 코드에서는,
Generic 타입의 클래스를 정의한 다음,
Person클래스를 String 또는 int타입으로 객체화 한 것을 볼 수 있습니다.
객체화 하는 시점에, 타입을 다르게 정한 것이지요.
(원래 Java에서는 기본 데이터타입은 올 수 없었는데,
Kotlin에서는 모두가 객체로 참조형이기에 그런 제한이 없습니다)
1-2. Generic 2개 사용하기
아래와 같이 Generic을 두 개 사용하는 것도 가능합니다.
마치 함수의 인자 넣는 것처럼 사용하는 것인데요.
세번째 코드에서 볼 수 있듯이,
추측할 수 있는 타입에 대해서는,
객체생성시에 Generic타입마저도 생략할 수 있습니다.
1-3. Generic Naming Convention
Generic에 사용되는 기호는 보통 T를 많이 사용하는데요.
T, S, U, V 를 차례로 사용하는 경우가 많습니다.
이 순서를 지키지 않는다고 컴파일 에러가 나지는 않고요.
그냥 K가 Key, V가 Value, R이 Receiver로 사용되는 것과 같이,
관습적인 사용일 뿐입니다.
2. Function에서의 Generic
Function에서도 Generic 사용이 가능한데요.
Generic을 붙이는 위치가,
Java와는 다르므로 주의가 필요합니다.
Class에서와 마찬가지로 추정할 수 있는 타입에 대해서는,
아래와 같이,
호출시 생략할 수 있습니다.
이번에는 조금 복잡해보이지만,
소스코드를 한번 보겠습니다.
<T, R>이라는 두개의 Generic타입을 사용하였는데요.
인자로 T타입의 Generic을 받아서,
Result클래스로 Wrapping된 R타입의 값을 반환해 주는 군요.
온통 Generic 타입이네요.
Generic을 모르면 이 짧은 코드를 이해하기 어렵습니다.
3. Generic에 있어서의 제한
Generic을 사용하면서 생기는 장점은,
Class사용시에 타입을 정의할 수 있다는 것 인데요.
다양한 타입을 정의할 수 있도록 하면서도,
그 범위를 제한하는 방법들이 있습니다.
3-1. In/Out
만약 Java에서 wildcard type의 Generic을 사용해 보았다면,
Generic의 In그리고 Out에 대해서 좀 더 이해하기 쉬울텐데요.
잠시 Java의 Wildcard와 비교해 가며 알아볼까요.
<E>라는 타입의 Generic은 읽기쓰기 모두에 적용될 수 있는 타입인데요.
반면에 와일드카드(?)를 사용하는 Java에서의 아래의 두가지 타입을 사용하면,
읽고 쓰는데 제한을 할 수 있습니다.
A. Java에서의 <? extends E> 와 Kotlin에서의 out
Read(읽기) 전용. E 또는 E의 자식타입들만 받아들여서 읽어들인다는 것 입니다.
참고로 producer와 consumer라는 개념을 사용하여서,
producer로부터 읽을 때 사용한다고 하기도 하는데,
upper bound(extends-bound)라고 표현하기도 합니다.
여기에 대응할 수 있는 Kotlin의 generic은 어떻게 표현할까요?
out 을 사용할 수 있습니다.
코드를 보면서 정리해 보겠습니다.
가장 흔하게 볼 수 있는 함수인 onRequestPermissionsResult를 보도록 하겠습니다.
유저에게 요청한 Permission의 결과에 따라서 결과를 주는 콜백함수인데요.
permissions를 보면, Out키워드를 볼 수 있습니다.
String과 그 자식들만 받아들이는 Array타입의 permission라고 하는 것을 알 수 있습니다.
한가지 더 보면, 아래는 List와 MutableList의 코드인데요.
E타입 혹은 그의 subtype을 read전용으로 가져올 수 있다고 Generic을 이용해서 기술되어 있습니다.
따라서, List에는 아이템을 추가할 수 있는 add()나 그와 유사한 메소드가 제공되지 않습니다.
반면 MutableList<E>는 E와 그 subtype으로 읽고 쓰기가 자유롭게 가능한 타입이지요.
당연히 add나 remove가 가능합니다.
B. <? super E >
위와는 반대로 write(쓰기)전용의 타입입니다.
E또는 E의 자식들만 쓸 수 있다는 것 이지요.
쓴다는 표현이 이해가 않가신다면,
List<? super String>의 경우와 같이,
Collection에 add혹은 put한다고 생각하면 좋을 것 같습니다.
producer와 consumer개념을 기준으로,
consumer에 write할 때 사용한다고 할 수 있습니다.
Kotlin에서는 in 키워드를 이용합니다.
이번에도 실제 소스코드에서 사용하는 예를 잠깐 보도록 하겠습니다.
Comparable인터페이스에는 in키워드가 사용되었는데요.
쓰기전용의 Generic타입T를 사용하였습니다.
C. in과 out의 구분
구분하기 어려울 때는 공식문서에 나오는 것처럼,
consumer in, producer out이라고 생각하면 좋을 것 같습니다.
즉 consumer에 write할 때는 in하고, producer로부터 read할 때는 out한다고 말이지요.
3-2. Upper Bounds
Generic타입뒤에 ":"을 붙이고 그 뒤에 타입을 넣어주면,
그 타입이거나 그 타입의 자식들만 올 수 있도록 할 수 있습니다.
이것을 UpperBounds라고 합니다.
아래 코드의 경우, 함수에 UpperBounds generic을 사용하였습니다.
R과 T 두개의 Generic타입중 T의 경우, R타입이거나 R타입의 자식타입들만 사용할 수 있도록 한 것이지요.
3-3. Where Bounds
UpperBounds이외에도 다른 제한을 두고자 한다면 where문을 사용하면 됩니다.
개인적으로는 아직 이정도를 사용해 보지는 못했지만,
우선은 아래와 같은 사용도 가능하다 정도로 알고 있으면 될 것 같습니다.
이상으로 Kotlin에서의 Generic사용의 기본적인 부분들에 대해서 정리해 보았습니다.
좀 더 필요한 내용이나 더욱 좋은 내용이 있다면 이 글에 추가하도록 하겠습니다.
'Android 개발 > Kotlin' 카테고리의 다른 글
Kotlin By 키워드에 대한 이해 # Property Delegate Pattern (1) | 2022.10.11 |
---|---|
Kotlin 확장 함수 및 Property 에 대한 정리 # Extension Functions (0) | 2022.10.04 |
vararg 로 가변 인자를 전달하는 방법 #Kotlin (0) | 2020.05.11 |
Kotlin으로 Android개발 하기 (0) | 2019.10.22 |
Kotlin Coroutine에 대한 정리 (0) | 2019.10.09 |
Scope Function 총정리 # Kotlin also let run with apply (0) | 2019.09.29 |
Kotlin을 배워보자 part5 # CompanionObject Object Expression and Declaration (1) | 2019.09.28 |
Kotlin을 배워보자 part4 (Data Class, Nested and Inner Class) (0) | 2019.09.27 |
Kotlin을 배워보자 part3 (Class, Constructor, 상속) (0) | 2019.09.26 |
Kotlin 을 배워보자 part2(if, when, map, for, while, array, list, singleOrNull, ranges, return with label) (0) | 2019.09.25 |
댓글