Kotlin에서 Java 호출하기
Kotlin은 Java 상호 운용성을 염두에 두고 설계되었습니다. 기존 Java 코드는 Kotlin에서 자연스러운 방식으로 호출될 수 있으며, Kotlin 코드 또한 Java에서 상당히 원활하게 사용될 수 있습니다. 이 섹션에서는 Kotlin에서 Java 코드를 호출하는 것에 대한 몇 가지 세부 정보를 설명합니다.
거의 모든 Java 코드는 문제없이 사용할 수 있습니다:
import java.util.*
fun demo(source: List<Int>) {
val list = ArrayList<Int>()
// 'for' 루프는 Java 컬렉션에서 작동합니다:
for (item in source) {
list.add(item)
}
// 연산자 규칙 또한 작동합니다:
for (i in 0..source.size - 1) {
list[i] = source[i] // get 및 set이 호출됩니다
}
}
게터 및 세터
Java의 게터(getter) 및 세터(setter) 규칙을 따르는 메서드(이름이 get
으로 시작하는 인자 없는 메서드와 이름이 set
으로 시작하는 단일 인자 메서드)는 Kotlin에서 프로퍼티로 표현됩니다. 이러한 프로퍼티는 _합성 프로퍼티_라고도 불립니다. Boolean
접근자 메서드(게터의 이름이 is
로 시작하고 세터의 이름이 set
으로 시작하는 경우)는 게터 메서드와 동일한 이름을 가진 프로퍼티로 표현됩니다.
import java.util.Calendar
fun calendarDemo() {
val calendar = Calendar.getInstance()
if (calendar.firstDayOfWeek == Calendar.SUNDAY) { // getFirstDayOfWeek() 호출
calendar.firstDayOfWeek = Calendar.MONDAY // setFirstDayOfWeek() 호출
}
if (!calendar.isLenient) { // isLenient() 호출
calendar.isLenient = true // setLenient() 호출
}
}
위 calendar.firstDayOfWeek
는 합성 프로퍼티의 예시입니다.
참고로, Java 클래스에 세터만 있는 경우 Kotlin에서는 세터 전용 프로퍼티를 지원하지 않기 때문에 프로퍼티로 보이지 않습니다.
Java 합성 프로퍼티 참조
Kotlin 1.8.20부터 Java 합성 프로퍼티에 대한 참조를 생성할 수 있습니다. 다음 Java 코드를 살펴보세요:
public class Person {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public int getAge() {
return age;
}
}
Kotlin은 항상 age
가 합성 프로퍼티인 person.age
를 작성하는 것을 허용했습니다. 이제 Person::age
와 person::age
에 대한 참조도 생성할 수 있습니다. name
에도 동일하게 적용됩니다.
val persons = listOf(Person("Jack", 11), Person("Sofie", 12), Person("Peter", 11))
persons
// Java 합성 프로퍼티에 대한 참조 호출:
.sortedBy(Person::age)
// Kotlin 프로퍼티 구문을 통해 Java 게터 호출:
.forEach { person -> println(person.name) }
Java 합성 프로퍼티 참조를 활성화하는 방법
이 기능을 활성화하려면 -language-version 2.1
컴파일러 옵션을 설정하세요. Gradle 프로젝트에서는 build.gradle(.kts)
에 다음을 추가하여 활성화할 수 있습니다:
tasks
.withType<org.jetbrains.kotlin.gradle.tasks.KotlinCompilationTask<*>>()
.configureEach {
compilerOptions
.languageVersion
.set(
org.jetbrains.kotlin.gradle.dsl.KotlinVersion.KOTLIN_2_1
)
}
tasks
.withType(org.jetbrains.kotlin.gradle.tasks.KotlinCompilationTask.class)
.configureEach {
compilerOptions.languageVersion
= org.jetbrains.kotlin.gradle.dsl.KotlinVersion.KOTLIN_2_1
}
NOTE
Kotlin 1.9.0 이전에는 이 기능을 활성화하려면 -language-version 1.9
컴파일러 옵션을 설정해야 했습니다.
void
를 반환하는 메서드
Java 메서드가 void
를 반환하는 경우, Kotlin에서 호출될 때 Unit
을 반환합니다. 만약 누군가 해당 반환 값을 사용하는 경우, 값 자체가 ( Unit
으로) 미리 알려져 있기 때문에 Kotlin 컴파일러에 의해 호출 위치에서 할당됩니다.
Kotlin 키워드인 Java 식별자 이스케이핑
일부 Kotlin 키워드는 Java에서 유효한 식별자입니다: in
, object
, is
등. Java 라이브러리가 메서드에 Kotlin 키워드를 사용하는 경우, 백틱(`
) 문자로 이스케이프하여 메서드를 여전히 호출할 수 있습니다:
foo.`is`(bar)
Null 안전성 및 플랫폼 타입
Java의 모든 참조는 null
일 수 있으며, 이는 Java에서 오는 객체에 대해 Kotlin의 엄격한 Null 안전성 요구사항을 비실용적으로 만듭니다. Java 선언의 타입은 Kotlin에서 특정 방식으로 처리되며 플랫폼 타입이라고 불립니다. 이러한 타입에 대해서는 Null 검사가 완화되어, 안전성 보장이 Java와 동일합니다 (자세한 내용은 아래 참조).
다음 예시를 살펴보세요:
val list = ArrayList<String>() // non-null (생성자 결과)
list.add("Item")
val size = list.size // non-null (원시 int)
val item = list[0] // 플랫폼 타입 추론됨 (일반 Java 객체)
플랫폼 타입 변수에 대해 메서드를 호출할 때, Kotlin은 컴파일 시점에 Null 가능성 오류를 발생시키지 않지만, Null이 전파되는 것을 방지하기 위해 Kotlin이 생성하는 Null-포인터 예외 또는 어설션 때문에 런타임에 호출이 실패할 수 있습니다:
item.substring(1) // 허용됨, item이 null인 경우 예외 발생
플랫폼 타입은 *표현 불가능(non-denotable)*합니다. 즉, 언어 내에서 명시적으로 작성할 수 없습니다. 플랫폼 값이 Kotlin 변수에 할당될 때, 타입 추론에 의존하거나 (위 예시에서 item
이 그랬듯이, 변수는 추론된 플랫폼 타입을 가집니다), 예상하는 타입을 선택할 수 있습니다 (널 허용 및 널 불허 타입 모두 허용됩니다):
val nullable: String? = item // 허용됨, 항상 작동함
val notNull: String = item // 허용됨, 런타임에 실패할 수 있음
널 불허 타입을 선택하는 경우, 컴파일러는 할당 시 어설션을 발생시킵니다. 이는 Kotlin의 널 불허 변수가 Null을 가지는 것을 방지합니다. 어설션은 플랫폼 값을 널 불허 값을 예상하는 Kotlin 함수에 전달할 때 및 다른 경우에도 발생합니다. 전반적으로, 컴파일러는 Null이 프로그램 전체에 걸쳐 멀리 전파되는 것을 방지하기 위해 최선을 다하지만, 제네릭 때문에 때로는 이를 완전히 제거하는 것이 불가능할 수도 있습니다.
플랫폼 타입 표기법
위에서 언급했듯이, 플랫폼 타입은 프로그램에서 명시적으로 언급될 수 없으므로 언어에 대한 구문이 없습니다. 그럼에도 불구하고, 컴파일러와 IDE는 때때로 이를 표시해야 하므로 (예: 오류 메시지 또는 파라미터 정보), 다음과 같은 기억하기 쉬운 표기법이 있습니다:
T!
는 "T
또는T?
"를 의미합니다.(Mutable)Collection<T>!
는 "JavaT
컬렉션은 가변(mutable)일 수도 아닐 수도 있고, Null 허용일 수도 아닐 수도 있습니다"를 의미합니다.Array<(out) T>!
는 "JavaT
배열(또는T
의 하위 타입)은 Null 허용일 수도 아닐 수도 있습니다"를 의미합니다.
Null 가능성 어노테이션
Null 가능성 어노테이션을 가진 Java 타입은 플랫폼 타입이 아니라 실제 Null 허용 또는 Null 불허 Kotlin 타입으로 표현됩니다. 컴파일러는 다음을 포함하여 여러 종류의 Null 가능성 어노테이션을 지원합니다:
- JetBrains (
org.jetbrains.annotations
패키지의@Nullable
및@NotNull
) - JSpecify (
org.jspecify.annotations
) - Android (
com.android.annotations
및android.support.annotations
) - JSR-305 (
javax.annotation
, 자세한 내용은 아래 참조) - FindBugs (
edu.umd.cs.findbugs.annotations
) - Eclipse (
org.eclipse.jdt.annotation
) - Lombok (
lombok.NonNull
) - RxJava 3 (
io.reactivex.rxjava3.annotations
)
특정 타입의 Null 가능성 어노테이션 정보에 따라 컴파일러가 Null 가능성 불일치를 보고할지 여부를 지정할 수 있습니다. 컴파일러 옵션 -Xnullability-annotations=@<package-name>:<report-level>
을 사용하세요. 인자에는 정규화된 Null 가능성 어노테이션 패키지와 다음 보고 수준 중 하나를 지정합니다:
ignore
: Null 가능성 불일치 무시warn
: 경고 보고strict
: 오류 보고.
Kotlin 컴파일러 소스 코드에서 지원되는 Null 가능성 어노테이션의 전체 목록을 참조하세요.
타입 인자 및 타입 파라미터 어노테이션 달기
제네릭 타입의 타입 인자 및 타입 파라미터에도 Null 가능성 정보를 제공하기 위해 어노테이션을 달 수 있습니다.
NOTE
이 섹션의 모든 예시는 org.jetbrains.annotations
패키지의 JetBrains Null 가능성 어노테이션을 사용합니다.
타입 인자
Java 선언에 있는 다음 어노테이션들을 고려해 보세요:
@NotNull
Set<@NotNull String> toSet(@NotNull Collection<@NotNull String> elements) { ... }
이들은 Kotlin에서 다음 시그니처를 결과로 가져옵니다:
fun toSet(elements: (Mutable)Collection<String>) : (Mutable)Set<String> { ... }
타입 인자에서 @NotNull
어노테이션이 누락되면, 대신 플랫폼 타입을 얻게 됩니다:
fun toSet(elements: (Mutable)Collection<String!>) : (Mutable)Set<String!> { ... }
Kotlin은 또한 기본 클래스 및 인터페이스의 타입 인자에 대한 Null 가능성 어노테이션을 고려합니다. 예를 들어, 아래에 제공된 시그니처를 가진 두 개의 Java 클래스가 있습니다:
public class Base<T> {}
public class Derived extends Base<@Nullable String> {}
Kotlin 코드에서 Base<String>
이 가정되는 곳에 Derived
의 인스턴스를 전달하면 경고가 발생합니다.
fun takeBaseOfNotNullStrings(x: Base<String>) {}
fun main() {
takeBaseOfNotNullStrings(Derived()) // 경고: nullability mismatch
}
Derived
의 상위 바운드는 Base<String?>
으로 설정되며, 이는 Base<String>
과 다릅니다.
Kotlin의 Java 제네릭에 대해 자세히 알아보세요.
타입 파라미터
기본적으로 Kotlin과 Java 모두에서 일반 타입 파라미터의 Null 가능성은 정의되지 않습니다. Java에서는 Null 가능성 어노테이션을 사용하여 이를 지정할 수 있습니다. Base
클래스의 타입 파라미터에 어노테이션을 달아보겠습니다:
public class Base<@NotNull T> {}
Base
를 상속할 때, Kotlin은 Null 불허 타입 인자 또는 타입 파라미터를 예상합니다. 따라서 다음 Kotlin 코드는 경고를 발생시킵니다:
class Derived<K> : Base<K> {} // 경고: K의 Null 가능성이 정의되지 않음
상위 바운드 K : Any
를 지정하여 수정할 수 있습니다.
Kotlin은 또한 Java 타입 파라미터의 바운드에 대한 Null 가능성 어노테이션을 지원합니다. Base
에 바운드를 추가해 보겠습니다:
public class BaseWithBound<T extends @NotNull Number> {}
Kotlin은 이를 다음과 같이 변환합니다:
class BaseWithBound<T : Number> {}
따라서 Null 허용 타입을 타입 인자 또는 타입 파라미터로 전달하면 경고가 발생합니다.
타입 인자 및 타입 파라미터에 어노테이션을 다는 것은 Java 8 이상을 대상으로 합니다. 이 기능은 Null 가능성 어노테이션이 TYPE_USE
타겟을 지원해야 합니다 (org.jetbrains.annotations
는 버전 15 이상에서 이를 지원합니다).
NOTE
Null 가능성 어노테이션이 TYPE_USE
타겟 외에 타입에 적용 가능한 다른 타겟을 지원하는 경우,
TYPE_USE
가 우선 순위를 가집니다. 예를 들어, @Nullable
에 TYPE_USE
와 METHOD
타겟이 모두 있다면, Java 메서드
시그니처 @Nullable String[] f()
는 Kotlin에서 fun f(): Array<String?>!
가 됩니다.
JSR-305 지원
JSR-305에 정의된 @Nonnull
어노테이션은 Java 타입의 Null 가능성을 나타내는 데 지원됩니다.
@Nonnull(when = ...)
값이 When.ALWAYS
인 경우, 어노테이션이 달린 타입은 Null 불허로 처리됩니다; When.MAYBE
및 When.NEVER
는 Null 허용 타입을 나타내며; When.UNKNOWN
은 해당 타입을 플랫폼 타입으로 강제합니다.
라이브러리는 JSR-305 어노테이션에 대해 컴파일될 수 있지만, 라이브러리 소비자를 위해 어노테이션 아티팩트(예: jsr305.jar
)를 컴파일 종속성으로 만들 필요는 없습니다. Kotlin 컴파일러는 클래스패스에 어노테이션이 없어도 라이브러리에서 JSR-305 어노테이션을 읽을 수 있습니다.
사용자 지정 Null 가능성 한정자 (KEEP-79)도 지원됩니다 (아래 참조).
타입 한정자 별칭
어노테이션 타입이 @TypeQualifierNickname
과 JSR-305 @Nonnull
(또는 @CheckForNull
과 같은 다른 별칭)으로 모두 어노테이션이 달린 경우, 해당 어노테이션 타입 자체는 정확한 Null 가능성을 검색하는 데 사용되며 해당 Null 가능성 어노테이션과 동일한 의미를 가집니다:
@TypeQualifierNickname
@Nonnull(when = When.ALWAYS)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyNonnull {
}
@TypeQualifierNickname
@CheckForNull // 다른 타입 한정자 별칭에 대한 별칭
@Retention(RetentionPolicy.RUNTIME)
public @interface MyNullable {
}
interface A {
@MyNullable String foo(@MyNonnull String x);
// Kotlin에서 (strict 모드): `fun foo(x: String): String?`
String bar(List<@MyNonnull String> x);
// Kotlin에서 (strict 모드): `fun bar(x: List<String>!): String!`
}
타입 한정자 기본값
@TypeQualifierDefault
는 어노테이션이 적용될 때, 어노테이션이 달린 요소의 범위 내에서 기본 Null 가능성을 정의하는 어노테이션을 도입할 수 있도록 합니다.
이러한 어노테이션 타입 자체는 @Nonnull
(또는 그 별칭)과 @TypeQualifierDefault(...)
모두에 하나 이상의 ElementType
값과 함께 어노테이션이 달려야 합니다:
ElementType.METHOD
: 메서드의 반환 타입용ElementType.PARAMETER
: 값 파라미터용ElementType.FIELD
: 필드용ElementType.TYPE_USE
: 타입 인자, 타입 파라미터의 상위 바운드 및 와일드카드 타입을 포함한 모든 타입용
기본 Null 가능성은 타입 자체에 Null 가능성 어노테이션이 달려 있지 않을 때 사용되며, 해당 기본값은 타입 사용과 일치하는 ElementType
를 가진 타입 한정자 기본 어노테이션으로 어노테이션이 달린 가장 안쪽의 감싸는 요소에 의해 결정됩니다.
@Nonnull
@TypeQualifierDefault({ElementType.METHOD, ElementType.PARAMETER})
public @interface NonNullApi {
}
@Nonnull(when = When.MAYBE)
@TypeQualifierDefault({ElementType.METHOD, ElementType.PARAMETER, ElementType.TYPE_USE})
public @interface NullableApi {
}
interface A {
String foo(String x); // fun foo(x: String?): String?
@NotNullApi // 인터페이스의 기본값 재정의
String bar(String x, @Nullable String y); // fun bar(x: String, y: String?): String
// `@NullableApi`가 `TYPE_USE` 요소 타입을 가지기 때문에 List<String> 타입 인자는 Null 허용으로 간주됩니다:
String baz(List<String> x); // fun baz(List<String?>?): String?
// 명시적으로 UNKNOWN으로 표시된 Null 가능성 어노테이션이 있으므로 `x` 파라미터의 타입은 플랫폼으로 유지됩니다:
String qux(@Nonnull(when = When.UNKNOWN) String x); // fun baz(x: String!): String?
}
NOTE
이 예시의 타입은 strict 모드가 활성화된 경우에만 적용됩니다; 그렇지 않으면 플랫폼 타입으로 유지됩니다.
@UnderMigration
어노테이션 및 컴파일러 구성 섹션을 참조하세요.
패키지 수준 기본 Null 가능성도 지원됩니다:
// FILE: test/package-info.java
@NonNullApi // 'test' 패키지의 모든 타입을 기본적으로 Null 불허로 선언
package test;
@UnderMigration 어노테이션
@UnderMigration
어노테이션(kotlin-annotations-jvm
이라는 별도 아티팩트로 제공됨)은 라이브러리 유지보수자가 Null 가능성 타입 한정자에 대한 마이그레이션 상태를 정의하는 데 사용할 수 있습니다.
@UnderMigration(status = ...)
의 status
값은 컴파일러가 Kotlin에서 어노테이션이 달린 타입의 부적절한 사용(예: @MyNullable
어노테이션이 달린 타입 값을 Null 불허로 사용)을 어떻게 처리하는지 지정합니다:
MigrationStatus.STRICT
: 어노테이션이 일반 Null 가능성 어노테이션처럼 작동하게 합니다. 즉, 부적절한 사용에 대해 오류를 보고하고 어노테이션이 달린 선언의 타입에 Kotlin에서 보이는 방식대로 영향을 미칩니다.MigrationStatus.WARN
: 부적절한 사용이 오류 대신 컴파일 경고로 보고되지만, 어노테이션이 달린 선언의 타입은 플랫폼으로 유지됩니다.MigrationStatus.IGNORE
: 컴파일러가 Null 가능성 어노테이션을 완전히 무시하게 합니다.
라이브러리 유지보수자는 @UnderMigration
상태를 타입 한정자 별칭과 타입 한정자 기본값 모두에 추가할 수 있습니다:
@Nonnull(when = When.ALWAYS)
@TypeQualifierDefault({ElementType.METHOD, ElementType.PARAMETER})
@UnderMigration(status = MigrationStatus.WARN)
public @interface NonNullApi {
}
// `@NonNullApi`가 `@UnderMigration(status = MigrationStatus.WARN)`으로 어노테이션이 달려있기 때문에
// 클래스의 타입은 Null 불허이지만, 경고만 보고됩니다.
@NonNullApi
public class Test {}
NOTE
Null 가능성 어노테이션의 마이그레이션 상태는 해당 타입 한정자 별칭에 의해 상속되지 않지만, 기본 타입 한정자에서의 사용에 적용됩니다.
기본 타입 한정자가 타입 한정자 별칭을 사용하고 둘 다 @UnderMigration
인 경우, 기본 타입 한정자의 상태가 사용됩니다.
컴파일러 구성
JSR-305 검사는 -Xjsr305
컴파일러 플래그에 다음 옵션(및 그 조합)을 추가하여 구성할 수 있습니다:
-Xjsr305={strict|warn|ignore}
:@UnderMigration
이 없는 어노테이션의 동작을 설정합니다. 사용자 지정 Null 가능성 한정자, 특히@TypeQualifierDefault
는 이미 많은 잘 알려진 라이브러리에 퍼져 있으며, 사용자는 JSR-305 지원을 포함하는 Kotlin 버전으로 업데이트할 때 원활하게 마이그레이션해야 할 수도 있습니다. Kotlin 1.1.60부터 이 플래그는@UnderMigration
이 없는 어노테이션에만 영향을 미칩니다.-Xjsr305=under-migration:{strict|warn|ignore}
:@UnderMigration
어노테이션의 동작을 재정의합니다. 사용자는 라이브러리의 마이그레이션 상태에 대해 다른 관점을 가질 수 있습니다. 공식 마이그레이션 상태가WARN
일 때 오류를 원하거나 그 반대의 경우도 가능하며, 마이그레이션을 완료할 때까지 일부에 대한 오류 보고를 연기할 수도 있습니다.-Xjsr305=@<fq.name>:{strict|warn|ignore}
: 단일 어노테이션의 동작을 재정의합니다. 여기서<fq.name>
은 어노테이션의 정규화된 클래스 이름입니다. 다른 어노테이션에 대해 여러 번 나타날 수 있습니다. 특정 라이브러리의 마이그레이션 상태를 관리하는 데 유용합니다.
strict
, warn
, ignore
값은 MigrationStatus
와 동일한 의미를 가지며, strict
모드만이 어노테이션이 달린 선언의 타입에 Kotlin에서 보이는 방식대로 영향을 미칩니다.
NOTE
참고: 내장 JSR-305 어노테이션인 @Nonnull
,
@CheckForNull
은 항상 활성화되어 있으며, -Xjsr305
플래그를 사용한 컴파일러 구성과 관계없이 Kotlin에서 어노테이션이 달린 선언의 타입에 영향을 미칩니다.
예를 들어, 컴파일러 인수에 -Xjsr305=ignore -Xjsr305=under-migration:ignore [email protected]:warn
를 추가하면 컴파일러가 @org.library.MyNullable
로 어노테이션이 달린 타입의 부적절한 사용에 대해 경고를 생성하고 다른 모든 JSR-305 어노테이션은 무시하게 됩니다.
기본 동작은 -Xjsr305=warn
과 동일합니다. strict
값은 실험적인 것으로 간주해야 합니다 (향후 더 많은 검사가 추가될 수 있습니다).
매핑된 타입
Kotlin은 일부 Java 타입을 특별하게 처리합니다. 이러한 타입은 Java에서 "있는 그대로" 로드되지 않고, 해당 Kotlin 타입으로 _매핑_됩니다. 매핑은 컴파일 시점에만 중요하며, 런타임 표현은 변경되지 않습니다. Java의 원시 타입은 해당 Kotlin 타입으로 매핑됩니다 (플랫폼 타입을 염두에 두고):
Java 타입 | Kotlin 타입 |
---|---|
byte | kotlin.Byte |
short | kotlin.Short |
int | kotlin.Int |
long | kotlin.Long |
char | kotlin.Char |
float | kotlin.Float |
double | kotlin.Double |
boolean | kotlin.Boolean |
일부 비원시 내장 클래스도 매핑됩니다:
Java 타입 | Kotlin 타입 |
---|---|
java.lang.Object | kotlin.Any! |
java.lang.Cloneable | kotlin.Cloneable! |
java.lang.Comparable | kotlin.Comparable! |
java.lang.Enum | kotlin.Enum! |
java.lang.annotation.Annotation | kotlin.Annotation! |
java.lang.CharSequence | kotlin.CharSequence! |
java.lang.String | kotlin.String! |
java.lang.Number | kotlin.Number! |
java.lang.Throwable | kotlin.Throwable! |
Java의 박싱된 원시 타입은 Null 허용 Kotlin 타입으로 매핑됩니다:
Java 타입 | Kotlin 타입 |
---|---|
java.lang.Byte | kotlin.Byte? |
java.lang.Short | kotlin.Short? |
java.lang.Integer | kotlin.Int? |
java.lang.Long | kotlin.Long? |
java.lang.Character | kotlin.Char? |
java.lang.Float | kotlin.Float? |
java.lang.Double | kotlin.Double? |
java.lang.Boolean | kotlin.Boolean? |
타입 파라미터로 사용된 박싱된 원시 타입은 플랫폼 타입으로 매핑된다는 점에 유의하세요. 예를 들어, List<java.lang.Integer>
는 Kotlin에서 List<Int!>
가 됩니다.
컬렉션 타입은 Kotlin에서 읽기 전용 또는 가변일 수 있으므로, Java의 컬렉션은 다음과 같이 매핑됩니다 (이 표의 모든 Kotlin 타입은 kotlin.collections
패키지에 있습니다):
Java 타입 | Kotlin 읽기 전용 타입 | Kotlin 가변 타입 | 로드된 플랫폼 타입 |
---|---|---|---|
Iterator<T> | Iterator<T> | MutableIterator<T> | (Mutable)Iterator<T>! |
Iterable<T> | Iterable<T> | MutableIterable<T> | (Mutable)Iterable<T>! |
Collection<T> | Collection<T> | MutableCollection<T> | (Mutable)Collection<T>! |
Set<T> | Set<T> | MutableSet<T> | (Mutable)Set<T>! |
List<T> | List<T> | MutableList<T> | (Mutable)List<T>! |
ListIterator<T> | ListIterator<T> | MutableListIterator<T> | (Mutable)ListIterator<T>! |
Map<K, V> | Map<K, V> | MutableMap<K, V> | (Mutable)Map<K, V>! |
Map.Entry<K, V> | Map.Entry<K, V> | MutableMap.MutableEntry<K,V> | (Mutable)Map.(Mutable)Entry<K, V>! |
Java의 배열은 아래에 언급된 바와 같이 매핑됩니다:
Java 타입 | Kotlin 타입 |
---|---|
int[] | kotlin.IntArray! |
String[] | kotlin.Array<(out) String!>! |
NOTE
이 Java 타입들의 정적 멤버는 Kotlin 타입의 동반 객체에서 직접 접근할 수 없습니다. 이들을 호출하려면 Java 타입의 정규화된 전체 이름을 사용하세요. 예를 들어 java.lang.Integer.toHexString(foo)
와 같이 사용합니다.
Kotlin의 Java 제네릭
Kotlin의 제네릭은 Java의 제네릭과 약간 다릅니다 (제네릭 참조). Java 타입을 Kotlin으로 가져올 때 다음 변환이 수행됩니다:
Java의 와일드카드는 타입 프로젝션으로 변환됩니다:
Foo<? extends Bar>
는Foo<out Bar!>!
가 됩니다.Foo<? super Bar>
는Foo<in Bar!>!
가 됩니다.
Java의 원시 타입은 스타 프로젝션으로 변환됩니다:
List
는List<*>!
즉List<out Any?>!
가 됩니다.
Java와 마찬가지로 Kotlin의 제네릭은 런타임에 유지되지 않습니다. 객체는 생성자에 전달된 실제 타입 인수에 대한 정보를 가지고 있지 않습니다. 예를 들어, ArrayList<Integer>()
는 ArrayList<Character>()
와 구별할 수 없습니다. 이로 인해 제네릭을 고려하는 is
검사를 수행하는 것이 불가능해집니다. Kotlin은 스타 프로젝션된 제네릭 타입에 대해서만 is
검사를 허용합니다:
if (a is List<Int>) // 오류: Int 리스트인지 실제로 확인할 수 없음
// 하지만
if (a is List<*>) // OK: 리스트 내용에 대한 보장은 없음
Java 배열
Kotlin의 배열은 Java와 달리 불변(invariant)입니다. 이는 Kotlin이 Array<String>
을 Array<Any>
에 할당하는 것을 허용하지 않아 잠재적인 런타임 실패를 방지함을 의미합니다. 서브클래스 배열을 슈퍼클래스 배열로 Kotlin 메서드에 전달하는 것도 금지되지만, Java 메서드의 경우 Array<(out) String>!
형태의 플랫폼 타입을 통해 허용됩니다.
Java 플랫폼에서는 박싱/언박싱 연산 비용을 피하기 위해 배열이 원시 데이터 타입과 함께 사용됩니다. Kotlin은 이러한 구현 세부 사항을 숨기므로 Java 코드와 인터페이스하려면 해결 방법이 필요합니다. 이 경우를 처리하기 위해 모든 원시 배열 타입(IntArray
, DoubleArray
, CharArray
등)에 대한 특수 클래스가 있습니다. 이들은 Array
클래스와 관련이 없으며 최대 성능을 위해 Java의 원시 배열로 컴파일됩니다.
인덱스(int 배열)를 받는 Java 메서드가 있다고 가정해 봅시다:
public class JavaArrayExample {
public void removeIndices(int[] indices) {
// 코드 삽입...
}
}
원시 값 배열을 전달하려면 Kotlin에서 다음을 수행할 수 있습니다:
val javaObj = JavaArrayExample()
val array = intArrayOf(0, 1, 2, 3)
javaObj.removeIndices(array) // 메서드에 int[] 전달
JVM 바이트코드로 컴파일할 때, 컴파일러는 배열에 대한 접근을 최적화하여 오버헤드가 발생하지 않도록 합니다:
val array = arrayOf(1, 2, 3, 4)
array[1] = array[1] * 2 // get() 및 set()에 대한 실제 호출은 생성되지 않음
for (x in array) { // 이터레이터가 생성되지 않음
print(x)
}
인덱스로 탐색할 때도 오버헤드가 발생하지 않습니다:
for (i in array.indices) { // 이터레이터가 생성되지 않음
array[i] += 2
}
마지막으로, in
검사에도 오버헤드가 없습니다:
if (i in array.indices) { // (i >= 0 && i < array.size)와 동일
print(array[i])
}
Java 가변 인자
Java 클래스는 때때로 가변 인자(varargs)로 인덱스에 대한 메서드 선언을 사용합니다:
public class JavaArrayExample {
public void removeIndicesVarArg(int... indices) {
// 코드 삽입...
}
}
이 경우 IntArray
를 전달하려면 스프레드 연산자 *
를 사용해야 합니다:
val javaObj = JavaArrayExample()
val array = intArrayOf(0, 1, 2, 3)
javaObj.removeIndicesVarArg(*array)
연산자
Java는 연산자 구문을 사용하는 것이 합리적인 메서드를 표시하는 방법이 없으므로, Kotlin은 올바른 이름과 시그니처를 가진 모든 Java 메서드를 연산자 오버로드 및 다른 규칙(invoke()
등)으로 사용할 수 있도록 허용합니다. 중위 호출 구문을 사용하여 Java 메서드를 호출하는 것은 허용되지 않습니다.
Checked 예외
Kotlin에서는 모든 예외가 unchecked이므로, 컴파일러가 어떤 예외도 강제로 catch하도록 하지 않습니다. 따라서 checked 예외를 선언하는 Java 메서드를 호출할 때 Kotlin은 아무것도 강제하지 않습니다:
fun render(list: List<*>, to: Appendable) {
for (item in list) {
to.append(item.toString()) // Java에서는 여기에 IOException을 catch하도록 요구할 것입니다
}
}
객체 메서드
Java 타입이 Kotlin으로 임포트될 때, java.lang.Object
타입의 모든 참조는 Any
로 변환됩니다. Any
는 플랫폼 특정적이지 않으므로, toString()
, hashCode()
, equals()
만을 멤버로 선언합니다. 따라서 java.lang.Object
의 다른 멤버를 사용할 수 있도록 Kotlin은 확장 함수를 사용합니다.
wait()
/notify()
wait()
및 notify()
메서드는 Any
타입의 참조에서 사용할 수 없습니다. 이들은 일반적으로 java.util.concurrent
를 선호하여 사용이 권장되지 않습니다. 정말로 이 메서드들을 호출해야 한다면 java.lang.Object
로 캐스팅할 수 있습니다:
(foo as java.lang.Object).wait()
getClass()
객체의 Java 클래스를 검색하려면 클래스 참조에서 java
확장 프로퍼티를 사용하세요:
val fooClass = foo::class.java
위 코드는 바운드 클래스 참조를 사용합니다. 또한 javaClass
확장 프로퍼티를 사용할 수도 있습니다:
val fooClass = foo.javaClass
clone()
clone()
을 오버라이드하려면 클래스가 kotlin.Cloneable
을 확장해야 합니다:
class Example : Cloneable {
override fun clone(): Any { ... }
}
Effective Java, 3rd Edition, 항목 13: clone을 신중하게 오버라이드하라를 잊지 마세요.
finalize()
finalize()
를 오버라이드하려면 override
키워드 없이 단순히 선언하기만 하면 됩니다:
class C {
protected fun finalize() {
// 종료 로직
}
}
Java 규칙에 따르면 finalize()
는 private
이어서는 안 됩니다.
Java 클래스 상속
Kotlin 클래스에서 최대 하나의 Java 클래스(원하는 만큼의 Java 인터페이스)가 슈퍼타입이 될 수 있습니다.
정적 멤버 접근
Java 클래스의 정적 멤버는 해당 클래스에 대한 "동반 객체"를 형성합니다. 이러한 "동반 객체"를 값으로 전달할 수는 없지만, 예를 들어 다음과 같이 명시적으로 멤버에 접근할 수 있습니다:
if (Character.isLetter(a)) { ... }
Kotlin 타입으로 매핑된 Java 타입의 정적 멤버에 접근하려면 Java 타입의 정규화된 전체 이름을 사용하세요: java.lang.Integer.bitCount(foo)
.
Java 리플렉션
Java 리플렉션은 Kotlin 클래스에서 작동하며 그 반대도 마찬가지입니다. 위에서 언급했듯이, java.lang.Class
를 통해 Java 리플렉션에 진입하려면 instance::class.java
, ClassName::class.java
또는 instance.javaClass
를 사용할 수 있습니다. 이 목적으로 ClassName.javaClass
를 사용하지 마세요. 이는 ClassName
의 동반 객체 클래스를 참조하며, 이는 ClassName.Companion::class.java
와 동일하고 ClassName::class.java
와는 다릅니다.
각 원시 타입에 대해 두 가지 다른 Java 클래스가 있으며, Kotlin은 둘 다 얻는 방법을 제공합니다. 예를 들어, Int::class.java
는 원시 타입 자체를 나타내는 클래스 인스턴스를 반환하며, 이는 Java의 Integer.TYPE
에 해당합니다. 해당 래퍼 타입의 클래스를 얻으려면 Java의 Integer.class
와 동등한 Int::class.javaObjectType
를 사용하세요.
그 외 지원되는 경우는 Kotlin 프로퍼티에 대한 Java 게터/세터 메서드 또는 백킹 필드를 얻는 것, Java 필드에 대한 KProperty
, KFunction
에 대한 Java 메서드 또는 생성자, 그리고 그 반대의 경우도 포함합니다.
SAM 변환
Kotlin은 Java 및 Kotlin 인터페이스 모두에 대한 SAM 변환을 지원합니다. Java에 대한 이러한 지원은 인터페이스 메서드의 파라미터 타입이 Kotlin 함수의 파라미터 타입과 일치하는 한, Kotlin 함수 리터럴이 단일 비기본(non-default) 메서드를 가진 Java 인터페이스의 구현으로 자동 변환될 수 있음을 의미합니다.
이를 사용하여 SAM 인터페이스의 인스턴스를 생성할 수 있습니다:
val runnable = Runnable { println("This runs in a runnable") }
...그리고 메서드 호출에서:
val executor = ThreadPoolExecutor()
// Java 시그니처: void execute(Runnable command)
executor.execute { println("이것은 스레드 풀에서 실행됩니다") }
Java 클래스에 함수형 인터페이스를 받는 여러 메서드가 있는 경우, 람다를 특정 SAM 타입으로 변환하는 어댑터 함수를 사용하여 호출해야 하는 메서드를 선택할 수 있습니다. 이러한 어댑터 함수는 필요할 때 컴파일러에 의해 생성됩니다:
executor.execute(Runnable { println("This runs in a thread pool") })
NOTE
SAM 변환은 인터페이스에만 작동하며, 추상 클래스에는 작동하지 않습니다. 해당 클래스가 단일 추상 메서드만 가지고 있더라도 마찬가지입니다.
Kotlin에서 JNI 사용
네이티브(C 또는 C++) 코드에 구현된 함수를 선언하려면 external
한정자로 표시해야 합니다:
external fun foo(x: Int): Double
나머지 절차는 Java와 완전히 동일하게 작동합니다.
프로퍼티 게터 및 세터를 external
로 표시할 수도 있습니다:
var myProperty: String
external get
external set
내부적으로, 이는 getMyProperty
와 setMyProperty
두 함수를 생성하며, 둘 다 external
로 표시됩니다.
Kotlin에서 Lombok 생성 선언 사용
Kotlin 코드에서 Java의 Lombok 생성 선언을 사용할 수 있습니다. 동일한 혼합 Java/Kotlin 모듈에서 이러한 선언을 생성하고 사용해야 하는 경우, Lombok 컴파일러 플러그인 페이지에서 그 방법을 배울 수 있습니다. 다른 모듈에서 이러한 선언을 호출하는 경우, 해당 모듈을 컴파일하기 위해 이 플러그인을 사용할 필요는 없습니다.