Я пытаюсь реализовать проверено Stack<E>
на основе Java-массива в Kotlin. Но у меня проблема с использованием KClass с моим универсальным типом параметра <E>
, который допускает нулевые значения.
Универсальные типы Java недоступны во время выполнения, но доступны типы массивов,Я хочу использовать эту функцию, чтобы во время выполнения была встроенная проверка типов.
Более подробную информацию о проверенном / непроверенном можно найти здесь https://stackoverflow.com/a/530289/10713249
interface Stack<E> {
fun push(elem: E)
fun pop(): E
}
class CheckedStack<E>(elementType: Class<E>, size: Int) : Stack<E> {
companion object {
inline fun <reified E> create(size: Int): CheckedStack<E> {
//**compile error here**
return CheckedStack(E::class.javaObjectType, size)
}
}
@Suppress("UNCHECKED_CAST")
private val array: Array<E?> = java.lang.reflect.Array.newInstance(elementType, size) as Array<E?>
private var index: Int = -1
override fun push(elem: E) {
check(index < array.size - 1)
array[++index] = elem
}
override fun pop(): E {
check(index >= 0);
@Suppress("UNCHECKED_CAST")
return array[index--] as E
}
}
Я ожидаю, что этот код будет работать так:
fun main() {
val intStack = CheckedStack.create<Int>(12) // Stack must store only Integer.class values
intStack.push(1); //[1]
intStack.push(2); //[1, 2]
val stackOfAny: Stack<Any?> = intStack as Stack<Any?>;
stackOfAny.push("str") // There should be a runtime error
}
Но у меня есть ошибка компиляции
Error:(39, 42) Kotlin: Type parameter bound for T in val <T : Any> KClass<T>.javaObjectType: Class<T>
is not satisfied: inferred type E is not a subtype of Any
Чтобы исправить это, мне нужно привязать параметр типа<E : Any>
но мне нужен стек, чтобы иметь возможность работать с обнуляемыми значениями <T : Any?>
. Как это исправить?
Почему KClass объявлен как KClass<T : Any>
, а не KClass<T : Any?>
?
UPD: Работает, если вместо E::class.java
E::class.javaObjectType
Потому чтосвойство val <T> KClass<T>.java: Class<T>
имеет тип param <T>
с аннотацией @Suppress("UPPER_BOUND_VIOLATED")
.
Но свойство val <T : Any> KClass<T>.javaObjectType: Class<T>
имеет тип <T : Any>
.
В моем случае Kotlin компилирует Int в Integer.class, а не в int (в моем случае). Но я не уверен, что это всегда будет работать.