将数组存储在变量中时出现 Kotlin ClassCastException
Kotlin ClassCastException when storing array in variable
我创建了一个 RollingWindow class 以便将固定数量的最新数据点存储在一个数组中。
class RollingWindow<T> (private val length: Int) {
private val window = Array<Any?>(length) {null}
private var count = 0
fun push(t: T) {
if (length == 0)
return
window[currentIndex()] = t
count++
}
@Suppress("UNCHECKED_CAST")
fun toArray(): Array<T> {
if (length == 0)
return arrayOf<Any>() as Array<T>
val firstHalf = window
.copyOfRange(currentIndex(), window.size)
.filterNotNull()
.toTypedArray()
val secondHalf = window
.copyOfRange(0, currentIndex())
.filterNotNull()
.toTypedArray()
val arr = arrayOf(*firstHalf, *secondHalf) as Array<T>
print(arr.contentToString())
//this works fine but for some reason the class cast exception is thrown from the unit test
return arr
}
override fun toString() = toArray().contentToString()
private fun currentIndex() = count % length
}
我写了一些单元测试并收到 ClassCastException
@Test
fun testRollingWindowNotFull() {
val doubleWindow = RollingWindow<Double>(5)
doubleWindow.push(2.5)
doubleWindow.push(6.8)
assertArrayEquals(arrayOf(2.5, 6.8), doubleWindow.toArray()) //this passes
val variableInWindow = RollingWindow<Double>(5)
variableInWindow.push(25.6)
variableInWindow.push(24.32)
val windowArray = variableInWindow.toArray() // ClassCastException only gets thrown here or whenever it's stored in a variable. If I use variableInWindow.toArray() inline it's fine, as shown in previous assertion
assertArrayEquals(arrayOf(25.6, 24.32), windowArray) // This never gets reached
}
在 运行 测试期间,我尝试将 Array 转换为 Array。在 RollingWindow class 中进行转换工作正常,没有错误,但我在单元测试中特别遇到错误。这是 StackTrace:
Sep 13, 2021 1:55:49 PM org.junit.platform.launcher.core.EngineDiscoveryOrchestrator lambda$logTestDescriptorExclusionReasons
INFO: 0 containers and 4 tests were Method or class mismatch
[Ljava.lang.Object;@7a8051ce[Ljava.lang.Object;@3ba12a08[Ljava.lang.Object;@725e196
class [Ljava.lang.Object; cannot be cast to class [Ljava.lang.Double; ([Ljava.lang.Object; and [Ljava.lang.Double; are in module java.base of loader 'bootstrap')
java.lang.ClassCastException: class [Ljava.lang.Object; cannot be cast to class [Ljava.lang.Double; ([Ljava.lang.Object; and [Ljava.lang.Double; are in module java.base of loader 'bootstrap')
at collections.RollingWindowTest.testRollingWindowNotFull(RollingWindowTest.kt:24)
与Kotlin中classes中的所有其他泛型不同,Arrayclass和Arrayclass只有一个具体化的类型,它是一个不变的类型。您永远无法将一种类型的数组成功转换为另一种类型的数组,除非您将其转换为具有协变或逆变类型。
我认为你的测试第一部分通过的唯一原因是必须有一个编译器优化,如果返回的对象被用作 Any
或 Array<out Any>
。 Java 定义的 assertArrayEquals
函数采用两个 Object[]
参数,它们映射到 Array<Any>
或 Array<out Any>
,也许在这种情况下它正在执行后者,因为它发现此函数中没有任何内容放入数组中。
因此作为优化的编译器可能会将您具体化的转换替换为 Array<Double>
的非具体化转换为 Array<out Double>
,这是一个安全的转换。
您可能需要考虑使用 List 或 MutableList 而不是 Array 来避免处理这些问题。
我创建了一个 RollingWindow class 以便将固定数量的最新数据点存储在一个数组中。
class RollingWindow<T> (private val length: Int) {
private val window = Array<Any?>(length) {null}
private var count = 0
fun push(t: T) {
if (length == 0)
return
window[currentIndex()] = t
count++
}
@Suppress("UNCHECKED_CAST")
fun toArray(): Array<T> {
if (length == 0)
return arrayOf<Any>() as Array<T>
val firstHalf = window
.copyOfRange(currentIndex(), window.size)
.filterNotNull()
.toTypedArray()
val secondHalf = window
.copyOfRange(0, currentIndex())
.filterNotNull()
.toTypedArray()
val arr = arrayOf(*firstHalf, *secondHalf) as Array<T>
print(arr.contentToString())
//this works fine but for some reason the class cast exception is thrown from the unit test
return arr
}
override fun toString() = toArray().contentToString()
private fun currentIndex() = count % length
}
我写了一些单元测试并收到 ClassCastException
@Test
fun testRollingWindowNotFull() {
val doubleWindow = RollingWindow<Double>(5)
doubleWindow.push(2.5)
doubleWindow.push(6.8)
assertArrayEquals(arrayOf(2.5, 6.8), doubleWindow.toArray()) //this passes
val variableInWindow = RollingWindow<Double>(5)
variableInWindow.push(25.6)
variableInWindow.push(24.32)
val windowArray = variableInWindow.toArray() // ClassCastException only gets thrown here or whenever it's stored in a variable. If I use variableInWindow.toArray() inline it's fine, as shown in previous assertion
assertArrayEquals(arrayOf(25.6, 24.32), windowArray) // This never gets reached
}
在 运行 测试期间,我尝试将 Array
Sep 13, 2021 1:55:49 PM org.junit.platform.launcher.core.EngineDiscoveryOrchestrator lambda$logTestDescriptorExclusionReasons
INFO: 0 containers and 4 tests were Method or class mismatch
[Ljava.lang.Object;@7a8051ce[Ljava.lang.Object;@3ba12a08[Ljava.lang.Object;@725e196
class [Ljava.lang.Object; cannot be cast to class [Ljava.lang.Double; ([Ljava.lang.Object; and [Ljava.lang.Double; are in module java.base of loader 'bootstrap')
java.lang.ClassCastException: class [Ljava.lang.Object; cannot be cast to class [Ljava.lang.Double; ([Ljava.lang.Object; and [Ljava.lang.Double; are in module java.base of loader 'bootstrap')
at collections.RollingWindowTest.testRollingWindowNotFull(RollingWindowTest.kt:24)
与Kotlin中classes中的所有其他泛型不同,Arrayclass和Arrayclass只有一个具体化的类型,它是一个不变的类型。您永远无法将一种类型的数组成功转换为另一种类型的数组,除非您将其转换为具有协变或逆变类型。
我认为你的测试第一部分通过的唯一原因是必须有一个编译器优化,如果返回的对象被用作 Any
或 Array<out Any>
。 Java 定义的 assertArrayEquals
函数采用两个 Object[]
参数,它们映射到 Array<Any>
或 Array<out Any>
,也许在这种情况下它正在执行后者,因为它发现此函数中没有任何内容放入数组中。
因此作为优化的编译器可能会将您具体化的转换替换为 Array<Double>
的非具体化转换为 Array<out Double>
,这是一个安全的转换。
您可能需要考虑使用 List 或 MutableList 而不是 Array 来避免处理这些问题。