1

具有类成员函数,例如:

private fun getData1(uuid:String): IData? {
    ...
}
private fun getData2(uuid:String): IData? {
    ...
}
private fun getData3(uuid:String): IData? {
    ...
}

并想放入一个函数引用数组:

var funArray = ArrayList<(uuid: String) -> IData?> (
     this::getData1, 
     this::getData2, 
     this::getData3)

它不编译:

None of the following functions can be called with the arguments 
supplied: 
public final fun <E> <init>(): kotlin.collections.ArrayList<(uuid: String) -> IData?> /* = java.util.ArrayList<(uuid: String) -> IData?> */ defined in kotlin.collections.ArrayList ...

如果这样做:

var funArray: ArrayList<(uuid: String) -> IData?> = ArrayList<(uuid: String) -> IData?>(3)

funArray[0] = this::getData1 //<== crash at here
funArray[1] = this::getData2
funArray[2] = this::getData3

异常崩溃

java.lang.IndexOutOfBoundsException: Index: 0, Size: 0

如何将函数引用放入数组中?

4

1 回答 1

4

第一次尝试失败,因为ArrayList没有构造函数采用(变量参数列表)值。

ArrayList您可以通过替换为listOf()(或者,如果您需要可变性, )获得几乎相同的效果mutableListOf(),因为它确实需要一个可变参数列表:

var functions = listOf<(uuid: String) -> IData?>(
    this::getData1, 
    this::getData2, 
    this::getData3)

这也许是最自然的解决方案。(但是,mutableListOf()只保证返回一个MutableList实现;它可能不是一个ArrayList.)

第二次尝试失败,因为它正在构建一个空列表。

(它使用的ArrayList 构造函数接受一个名为 的参数initialCapacity;它确保列表可以包含至少 3 个元素而无需重新分配其数组,但其初始大小为零。)

也许混淆是因为尽管您说您“想要放入一个函数引用数组”,但您创建的是一个List,而不是一个Array.

ArrayList该类是List接口的实现,它恰好在内部使用了一个数组。这遵循将实现类命名为的 Java 约定<Implementation><Interface>。)

如果您需要创建一个实际的数组,您可以arrayOf()在第一个示例中使用:

var functions = arrayOf<(uuid: String) -> IData?>(
    this::getData1, 
    this::getData2, 
    this::getData3)

列表在 Kotlin 中的使用可能比数组更广泛,因为它们更灵活。(您可以在具有不同特征的许多不同实现之间进行选择。它们与泛型配合得更好;例如,您可以创建List泛型类型的 a。您可以使它们不可变。当然,如果它们是可变的,它们可以增长和收缩。)

但是数组也有它们的位置,特别是如果性能很重要,您需要与使用数组的代码进行互操作,并且/或者大小是固定的。

于 2019-05-06T21:57:43.553 回答