type | layout | category | title | url |
---|---|---|---|---|
doc |
reference |
Java Interop |
Вызов Java из Kotlin |
Kotlin изначально проектировался с учетом необходимости взаимодействия с языком Java. Существующий код Java может быть вызван из Kotlin естественным способом, также как и код на Kotlin может быть использован в Java без особых проблем. В этом разделе мы опишем некоторые детали о вызове кода Java из Kotlin.
Почти любой код Java может быть использован без особых проблем:
import java.util.*
fun demo(source: List<Int>) {
val list = ArrayList<Int>()
// Циклы 'for' работают с коллекциями Java:
for (item in source) {
list.add(item)
}
// Operator conventions work as well:
for (i in 0..source.size - 1) {
list[i] = source[i] // вызваны get и set
}
}
Методы, которые следуют конвенциям Java для геттеров и сеттеров (методы без аргументов с именами, начинающимися на 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()
}
}
Имейте в виду, что если класс Java имеет только сеттер, он не будет виден как свойство в Kotlin, потому что Kotlin не поддерживает set-only свойства на данный момент.
Если метод Java возвращает void, он вернет Unit
при вызове из Kotlin.
Если по какой-либо причине кто-либо использует данный тип в своих целях, он будет присвоен в месте вызова
компилятором Kotlin, так как значение само по себе известно заранее (являясь Unit
).
Некоторые из ключевых слов Kotlin являются валидными идентификаторами в Java: in{: .keyword }, object{: .keyword }, is{: .keyword }, и т.д. Если Java-библиотека использует ключевые слова Kotlin в названиях методов, вы все еще можете вызывать метод используя экранирование с помощью обратной кавычки (`):
foo.`is`(bar)
Любая ссылка в Java может прнимать значение null{: .keyword }, что делает требования Kotlin по null-safety непрактичными для объектов приходящих из Java. Типы, декларируемые в 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 не обрабатывает ошибки отсутствия значения во время компиляции, но работа этих методов может вызвать ошибки на рантайме по причине NPE либо ассершена, генерируемого Kotlin для предотвращения всплытия null значений:
item.substring(1) // разрешается, может выбросить исключение, если item == null
Платформенные типы являются необозначаемыми, что значит, что их нельзя объявить непосредственно в языке.
Когда платформенное значение присваивается переменной в Kotlin, мы можем полагаться на автоматическое
определение типа (переменная получит подразумеваемый платформенный тип, так же как item
приведенная в примерах выше),
или же мы можем сами выбрать ожидаемый тип (разрешаются как nullable, так и non-null типы):
val nullable: String? = item // разрешается, всегда работает
val notNull: String = item // разрешается, может вызвать ошибку на рантайме
Если мы выбираем non-null тип, компилятор выбросит ассершен при присвоении значения. Это предотвратит non-null переменные Kotlin от хранения null-значений. Ассершены так же выбрасываются, когда мы передаем платформенные значения в функции Kotlin, ожидающие non-null значения. В целом, компилятор делает все возможное для предотвращения всплытия null-значений в глубине программы (и все же иногда возникают проблемы при использовании дженериков).
Как упоминалось выше, платформенные типы не могут быть явно указаны в программе, так что не существует синтаксиса для их обозначения в языке. И все же иногда компилятору и IDE нужно их обозначать (в сообщениях об ошибках, информации о параметрах, и т.д.). Для подобных случаев у нас имеется мнемоническая нотация для них:
T!
означает "T
илиT?
",(Mutable)Collection<T>!
означает "коллекция Java типаT
может быть (не)изменяемой, может быть nullable или нет",Array<(out) T>!
означает "массив Java типаT
(или подтипаT
), nullable или нет"
Типы Java, которые имеют аннотации допустимости null-значений представлены не как платформенные типы, а как реальные nullable или non-null типы Kotlin. Компилятор поддерживает несколько стандартов аннотаций, включая:
- JetBrains
(
@Nullable
и@NotNull
из пакетаorg.jetbrains.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
).
Вы можете найти полный список в Исходных текстах компилятора Kotlin.
Аннотация @Nonnull
объявленная
в JSR-305 поддерживается для использования с типами Java.
If the @Nonnull(when = ...)
value is When.ALWAYS
, the annotated type is treated as non-null; When.MAYBE
and
When.NEVER
denote a nullable type; and When.UNKNOWN
forces the type to be platform one.
A library can be compiled against the JSR-305 annotations, but there's no need to make the annotations artifact (e.g. jsr305.jar
)
a compile dependency for the library consumers. The Kotlin compiler can read the JSR-305 annotations from a library without the annotations
present on the classpath.
Начиная с версии Kotlin 1.1.50, custom nullability qualifiers (KEEP-79) are also supported (see below).
If an annotation type is annotated with both
@TypeQualifierNickname
and JSR-305 @Nonnull
(or its another nickname, such as @CheckForNull
), then the annotation type is itself used for
retrieving precise nullability and has the same meaning as that nullability annotation:
@TypeQualifierNickname
@Nonnull(when = When.ALWAYS)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyNonnull {
}
@TypeQualifierNickname
@CheckForNull // a nickname to another type qualifier nickname
@Retention(RetentionPolicy.RUNTIME)
public @interface MyNullable {
}
interface A {
@MyNullable String foo(@MyNonnull String x);
// in Kotlin (strict mode): `fun foo(x: String): String?`
String bar(List<@MyNonnull String> x);
// in Kotlin (strict mode): `fun bar(x: List<String>!): String!`
}
@TypeQualifierDefault
allows introducing annotations that, when being applied, define the default nullability within the scope of the annotated
element.
Such annotation type should itself be annotated with both @Nonnull
(or its nickname) and @TypeQualifierDefault(...)
with one or more ElementType
values:
ElementType.METHOD
for return types of methods;ElementType.PARAMETER
for value parameters;ElementType.FIELD
for fields; andElementType.TYPE_USE
(since 1.1.60) for any type including type arguments, upper bounds of type parameters and wildcard types.
The default nullability is used when a type itself is not annotated by a nullability annotation, and the default is
determined by the innermost enclosing element annotated with a type qualifier default annotation with the
ElementType
matching the type usage.
@Nonnull
@TypeQualifierDefault({ElementType.METHOD, ElementType.PARAMETER})
public @interface NonNullApi {
}
@Nonnull(when = When.MAYBE)
@TypeQualifierDefault({ElementType.METHOD, ElementType.PARAMETER, ElementType.TYPE_USE})
public @interface NullableApi {
}
@NullableApi
interface A {
String foo(String x); // fun foo(x: String?): String?
@NotNullApi // overriding default from the interface
String bar(String x, @Nullable String y); // fun bar(x: String, y: String?): String
// The List<String> type argument is seen as nullable because of `@NullableApi`
// having the `TYPE_USE` element type:
String baz(List<String> x); // fun baz(List<String?>?): String?
// The type of `x` parameter remains platform because there's an explicit
// UNKNOWN-marked nullability annotation:
String qux(@Nonnull(when = When.UNKNOWN) String x); // fun baz(x: String!): String?
}
Note: the types in this example only take place with the strict mode enabled, otherwise, the platform types remain. See the
@UnderMigration
annotation and Compiler configuration sections.
Package-level default nullability is also supported:
// FILE: test/package-info.java
@NonNullApi // declaring all types in package 'test' as non-nullable by default
package test;
The @UnderMigration
annotation (provided in a separate artifact kotlin-annotations-jvm
) can be used by library
maintainers to define the migration status for the nullability type qualifiers.
The status value in @UnderMigration(status = ...)
specifies how the compiler treats inappropriate usages of the
annotated types in Kotlin (e.g. using a @MyNullable
-annotated type value as non-null):
-
MigrationStatus.STRICT
makes annotation work as any plain nullability annotation, i.e. report errors for the inappropriate usages and affect the types in the annotated declarations as they are seen in Kotlin; -
with
MigrationStatus.WARN
, the inappropriate usages are reported as compilation warnings instead of errors, but the types in the annotated declarations remain platform; and -
MigrationStatus.IGNORE
makes the compiler ignore the nullability annotation completely.
A library maintainer can add @UnderMigration
status to both type qualifier nicknames and type qualifier defaults:
@Nonnull(when = When.ALWAYS)
@TypeQualifierDefault({ElementType.METHOD, ElementType.PARAMETER})
@UnderMigration(status = MigrationStatus.WARN)
public @interface NonNullApi {
}
// The types in the class are non-null, but only warnings are reported
// because `@NonNullApi` is annotated `@UnderMigration(status = MigrationStatus.WARN)`
@NonNullApi
public class Test {}
Note: the migration status of a nullability annotation is not inherited by its type qualifier nicknames but is applied to its usages in default type qualifiers.
If a default type qualifier uses a type qualifier nickname and they are both @UnderMigration
, the status
from the default type qualifier is used.
The JSR-305 checks can be configured by adding the -Xjsr305
compiler flag with the following options (and their combination):
-
-Xjsr305={strict|warn|ignore}
to set up the behavior for non-@UnderMigration
annotations. Custom nullability qualifiers, especially@TypeQualifierDefault
, are already spread among many well-known libraries, and users may need to migrate smoothly when updating to the Kotlin version containing JSR-305 support. Since Kotlin 1.1.60, this flag only affects non-@UnderMigration
annotations. -
-Xjsr305=under-migration:{strict|warn|ignore}
(since 1.1.60) to override the behavior for the@UnderMigration
annotations. Users may have different view on the migration status for the libraries: they may want to have errors while the official migration status isWARN
, or vice versa, they may wish to postpone errors reporting for some until they complete their migration. -
-Xjsr305=@<fq.name>:{strict|warn|ignore}
(since 1.1.60) to override the behavior for a single annotation, where<fq.name>
is the fully qualified class name of the annotation. May appear several times for different annotations. This is useful for managing the migration state for a particular library.
The strict
, warn
and ignore
values have the same meaning as those of MigrationStatus
, and only the strict
mode affects the types in the annotated declarations as they are seen in Kotlin.
Note: the built-in JSR-305 annotations
@Nonnull
,@Nullable
and@CheckForNull
are always enabled and affect the types of the annotated declarations in Kotlin, regardless of compiler configuration with the-Xjsr305
flag.
For example, adding -Xjsr305=ignore -Xjsr305=under-migration:ignore [email protected]:warn
to the
compiler arguments makes the compiler generate warnings for inappropriate usages of types annotated by
@org.library.MyNullable
and ignore all other JSR-305 annotations.
For kotlin versions 1.1.50+/1.2, the default behavior is the same to -Xjsr305=warn
. The
strict
value should be considered experimental (more checks may be added to it in the future).
Kotlin treats some Java types specially. Such types are not loaded from Java "as is", but are mapped to corresponding Kotlin types. The mapping only matters at compile time, the runtime representation remains unchanged. Java's primitive types are mapped to corresponding Kotlin types (keeping platform types in mind):
Тип 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 |
{:.zebra} |
Some non-primitive built-in classes are also mapped:
Тип 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 |
kotlin.Annotation! |
java.lang.Deprecated |
kotlin.Deprecated! |
java.lang.CharSequence |
kotlin.CharSequence! |
java.lang.String |
kotlin.String! |
java.lang.Number |
kotlin.Number! |
java.lang.Throwable |
kotlin.Throwable! |
{:.zebra} |
Java's boxed primitive types are mapped to nullable Kotlin types:
Java type | Kotlin type |
---|---|
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? |
{:.zebra} |
Note that a boxed primitive type used as a type parameter is mapped to a platform type:
for example, List<java.lang.Integer>
becomes a List<Int!>
in Kotlin.
Collection types may be read-only or mutable in Kotlin, so Java's collections are mapped as follows
(all Kotlin types in this table reside in the package kotlin.collections
):
Java type | Kotlin read-only type | Kotlin mutable type | Loaded platform type |
---|---|---|---|
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>! |
{:.zebra} |
Java's arrays are mapped as mentioned below:
Java type | Kotlin type |
---|---|
int[] |
kotlin.IntArray! |
String[] |
kotlin.Array<(out) String>! |
{:.zebra} |
Kotlin's generics are a little different from Java's (see Generics). When importing Java types to Kotlin we perform some conversions:
-
Java's wildcards are converted into type projections,
Foo<? extends Bar>
becomesFoo<out Bar!>!
,Foo<? super Bar>
becomesFoo<in Bar!>!
;
-
Java's raw types are converted into star projections,
List
becomesList<*>!
, i.e.List<out Any?>!
.
Like Java's, Kotlin's generics are not retained at runtime, i.e. objects do not carry information about actual type arguments passed to their constructors,
i.e. ArrayList<Integer>()
is indistinguishable from ArrayList<Character>()
.
This makes it impossible to perform is{: .keyword }-checks that take generics into account.
Kotlin only allows is{: .keyword }-checks for star-projected generic types:
if (a is List<Int>) // Error: cannot check if it is really a List of Ints
// but
if (a is List<*>) // OK: no guarantees about the contents of the list
Arrays in Kotlin are invariant, unlike Java. This means that Kotlin does not let us assign an Array<String>
to an Array<Any>
,
which prevents a possible runtime failure. Passing an array of a subclass as an array of superclass to a Kotlin method is also prohibited,
but for Java methods this is allowed (through platform types of the form Array<(out) String>!
).
Arrays are used with primitive datatypes on the Java platform to avoid the cost of boxing/unboxing operations.
As Kotlin hides those implementation details, a workaround is required to interface with Java code.
There are specialized classes for every type of primitive array (IntArray
, DoubleArray
, CharArray
, and so on) to handle this case.
They are not related to the Array
class and are compiled down to Java's primitive arrays for maximum performance.
Suppose there is a Java method that accepts an int array of indices:
public class JavaArrayExample {
public void removeIndices(int[] indices) {
// здесь ваш код...
}
}
To pass an array of primitive values you can do the following in Kotlin:
val javaObj = JavaArrayExample()
val array = intArrayOf(0, 1, 2, 3)
javaObj.removeIndices(array) // passes int[] to method
При компиляции в байт-код JVM, компилятор оптимизирует доступ к массивам, таким образом, что отсутствуют какие-либо накладные расходы:
val array = arrayOf(1, 2, 3, 4)
array[x] = array[x] * 2 // no actual calls to get() and set() generated
for (x in array) { // не создается итератор
print(x)
}
Even when we navigate with an index, it does not introduce any overhead:
for (i in array.indices) { // не создается итератор
array[i] += 2
}
Finally, in{: .keyword }-checks have no overhead either:
if (i in array.indices) { // same as (i >= 0 && i < array.size)
print(array[i])
}
Java classes sometimes use a method declaration for the indices with a variable number of arguments (varargs):
public class JavaArrayExample {
public void removeIndicesVarArg(int... indices) {
// здесь ваш код...
}
}
In that case you need to use the spread operator *
to pass the IntArray
:
val javaObj = JavaArrayExample()
val array = intArrayOf(0, 1, 2, 3)
javaObj.removeIndicesVarArg(*array)
It's currently not possible to pass null{: .keyword } to a method that is declared as varargs.
Since Java has no way of marking methods for which it makes sense to use the operator syntax, Kotlin allows using any
Java methods with the right name and signature as operator overloads and other conventions (invoke()
etc.)
Calling Java methods using the infix call syntax is not allowed.
В Kotlin все исключения непроверяемые (unchecked), это значит, что компилятор не будет заставлять Вас их обрабатывать. Так, когда Вы вызываете Java метод, который может выбросить проверяемое (checked) исключение, Kotlin не заставит Вас его обработать:
fun render(list: List<*>, to: Appendable) {
for (item in list) {
to.append(item.toString()) // Java потребует здесь обработки IOException
}
}
When Java types are imported into Kotlin, all the references of the type java.lang.Object
are turned into Any
.
Since Any
is not platform-specific, it only declares toString()
, hashCode()
and equals()
as its members,
so to make other members of java.lang.Object
available, Kotlin uses extension functions.
Effective Java Item 69 kindly suggests to prefer concurrency utilities to wait()
and notify()
.
Thus, these methods are not available on references of type Any
.
If you really need to call them, you can cast to java.lang.Object
:
(foo as java.lang.Object).wait()
To retrieve the Java class of an object, use the java
extension property on a class reference:
val fooClass = foo::class.java
The code above uses a bound class reference, which is supported since Kotlin 1.1. You can also use the javaClass
extension property:
val fooClass = foo.javaClass
To override clone()
, your class needs to extend kotlin.Cloneable
:
class Example : Cloneable {
override fun clone(): Any { ... }
}
Do not forget about Effective Java, Item 11: Override clone judiciously.
To override finalize()
, all you need to do is simply declare it, without using the override{:.keyword} keyword:
class C {
protected fun finalize() {
// finalization logic
}
}
According to Java's rules, finalize()
must not be private{: .keyword }.
At most one Java class (and as many Java interfaces as you like) can be a supertype for a class in Kotlin.
Static members of Java classes form "companion objects" for these classes. We cannot pass such a "companion object" around as a value, but can access the members explicitly, for example:
if (Character.isLetter(a)) {
// ...
}
Java reflection works on Kotlin classes and vice versa. As mentioned above, you can use instance::class.java
,
ClassName::class.java
or instance.javaClass
to enter Java reflection through java.lang.Class
.
Other supported cases include acquiring a Java getter/setter method or a backing field for a Kotlin property, a KProperty
for a Java field, a Java method or constructor for a KFunction
and vice versa.
Как и Java 8, Kotlin поддерживает SAM (Single Abstract Method) преобразования. Это означает, что литералы функций в Kotlin могут быть автоматически преобразованы к реализации Java интерфейсов с одним абстрактным методом, если типы параметров функции в Kotlin совпадают с типами параметров метода интерфеса.
Вы можете использовать это для создания объектов SAM интерфейсов:
val runnable = Runnable { println("This runs in a runnable") }
...и в вызовах метода:
val executor = ThreadPoolExecutor()
// Java signature: void execute(Runnable command)
executor.execute { println("This runs in a thread pool") }
Если Java класс имеет несколько методов, принимающих функциональные интерфейсы, Вы можете выбрать тот, который Вам нужен, используя адаптер функции, который преобразует лямбду в необходимый SAM тип. Эти адаптеры также генерируются компилятором, когда это необходимо:
executor.execute(Runnable { println("This runs in a thread pool") })
Обратите внимание, что SAM преобразования работают только для интерфейсов, не для абстрактных классов, даже если у них всего один абстрактный метод.
Также обратите внимание, что SAM преобразования работают только для Java совместимости; поскольку в Kotlin есть функциональные типы, автоматическое преобразование функий в реализацию Kotlin интерфейсов не требуется, поэтому не поддерживается.
Чтобы объявить функцию, реализованную на нативном (C или C++) коде, Вам нужно пометить ее модификатором external
:
external fun foo(x: Int): Double
В остальном, процедура работы аналогична Java.