将数组存储在变量中时出现 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只有一个具体化的类型,它是一个不变的类型。您永远无法将一种类型的数组成功转换为另一种类型的数组,除非您将其转换为具有协变或逆变类型。

我认为你的测试第一部分通过的唯一原因是必须有一个编译器优化,如果返回的对象被用作 AnyArray<out Any>。 Java 定义的 assertArrayEquals 函数采用两个 Object[] 参数,它们映射到 Array<Any>Array<out Any>,也许在这种情况下它正在执行后者,因为它发现此函数中没有任何内容放入数组中。

因此作为优化的编译器可能会将您具体化的转换替换为 Array<Double> 的非具体化转换为 Array<out Double>,这是一个安全的转换。

您可能需要考虑使用 List 或 MutableList 而不是 Array 来避免处理这些问题。