2

我想定义一个扩展,enum class它接受与枚举常量相同顺序的值列表并输出EnumMap.

相反,我能够做的是使用作为输入 List<V>的键数组创建和扩展对象。Enum<K>.values()

似乎困难在于enum class本身不是具有.values()方法的对象。所以也许我的要求是不明智的。

这是一个使用列表扩展方法的工作示例。

import java.util.*

enum class Shoes(
    val title: String,
    val weight: Double) {
    WORKBOOT("tough", 11.0),
    SNEAKER("fast", 6.0),
    SLIPPER("soft", 3.0);
}

fun main(args: Array<String>) {
    val prices= listOf(11.5,8.2,3.5)
    val map = prices.enumMapOf(Shoes.values())
    map.print()
}

inline fun <reified K : Enum<K>, V> List<V>.enumMapOf(keys:Array<K>): EnumMap<K, V> {
    val map = EnumMap<K, V>(K::class.java)
    keys.forEachIndexed{i,k-> map[k]=this[i]}
    return map
}

fun <K:Enum<K>,V> EnumMap<K,V>.print() {
    this.forEach{k,v-> println("%d: %s --> %.2f".format(k.ordinal,k.name,v)) }
}
4

3 回答 3

6

我认为您缺少的主要内容是enumValues可以为您提供通用枚举值的方法。

这将允许您执行此操作(我也重新安排了创建的代码,EnumMap如果这看起来太复杂,您可以坚持自己的解决方案):

inline fun <reified K : Enum<K>, V> Enum<K>.mappedTo(values: List<V>): EnumMap<K, V> {
    return enumValues<K>().zip(values).toMap(EnumMap(K::class.java))
}

当然,现在这个问题是调用语法——这个函数必须像这样在 enum 的实例上调用:

Shoes.WORKBOOT.mappedTo(listOf(11.5, 8.2, 3.5))

这没有多大意义。打这个电话会更好:

Shoes.mappedTo(listOf(11.5, 8.2, 3.5))

但据我所知,没有办法实现这种语法,因为你不能向通用枚举的伴随对象添加扩展,然后期望在特定枚举的伴随对象上调用它——那些不继承从彼此这样。

因此,也许您可​​以使用常规函数而不是创建扩展,并从泛型参数而不是接收器获取枚举的类型:

inline fun <reified K : Enum<K>, V> mapEnumTo(values: List<V>): EnumMap<K, V> {
    return enumValues<K>().zip(values).toMap(EnumMap(K::class.java))
}

mapEnumTo<Shoes, Double>(listOf(11.5, 8.2, 3.5))

您还必须在此处指定键的类型,因为您不能有一个显式的泛型类型参数,然后是另一个隐式的类型参数。但我认为这是非常好的语法,因为您可以看到您正在创建的映射的键和值类型。

于 2018-06-21T18:53:19.113 回答
5

作为Double从@zsmb13 的答案中消除显式的一种解决方法,您可以借用一个标准的 Scala 技巧,将其拆分为具有不同类型参数列表的两个方法调用,其中一个是显式的,另一个是推断的:

interface EnumMapper<K : Enum<K>> {
    operator fun <V> invoke(values: List<V>): EnumMap<K, V>
}

inline fun <reified K : Enum<K>> mapEnumTo() = object : EnumMapper<K> {
    override operator fun <V> invoke(values: List<V>): EnumMap<K, V> {
        return enumValues<K>().zip(values).toMap(EnumMap(K::class.java))
    }
}

mapEnumTo<Shoes>()(listOf(11.5, 8.2, 3.5))
于 2018-06-21T19:13:59.323 回答
-1

在枚举本身: private val shoeMap by lazy { values().associateBy { it.title } }

于 2019-11-01T01:07:14.613 回答