Mi sono ritrovato a fare la ricerca inversa da codice personalizzato, codificato a mano, valore un paio di volte e sono arrivato con il seguente approccio.
Fai enum
s implementano un'interfaccia comune:
interface Codified<out T : Serializable> {
val code: T
}
enum class Alphabet(val value: Int) : Codified<Int> {
A(1),
B(2),
C(3);
override val code = value
}
Questa interfaccia (per quanto strano il nome è :)) segna un certo valore come il codice esplicito. L'obiettivo è quello di essere in grado di scrivere:
val a = Alphabet::class.decode(1) //Alphabet.A
val d = Alphabet::class.tryDecode(4) //null
che può essere facilmente raggiunto con il seguente codice:
interface Codified<out T : Serializable> {
val code: T
object Enums {
private val enumCodesByClass = ConcurrentHashMap<Class<*>, Map<Serializable, Enum<*>>>()
inline fun <reified T, TCode : Serializable> decode(code: TCode): T where T : Codified<TCode>, T : Enum<*> {
return decode(T::class.java, code)
}
fun <T, TCode : Serializable> decode(enumClass: Class<T>, code: TCode): T where T : Codified<TCode> {
return tryDecode(enumClass, code) ?: throw IllegalArgumentException("No $enumClass value with code == $code")
}
inline fun <reified T, TCode : Serializable> tryDecode(code: TCode): T? where T : Codified<TCode> {
return tryDecode(T::class.java, code)
}
@Suppress("UNCHECKED_CAST")
fun <T, TCode : Serializable> tryDecode(enumClass: Class<T>, code: TCode): T? where T : Codified<TCode> {
val valuesForEnumClass = enumCodesByClass.getOrPut(enumClass as Class<Enum<*>>, {
enumClass.enumConstants.associateBy { (it as T).code }
})
return valuesForEnumClass[code] as T?
}
}
}
fun <T, TCode> KClass<T>.decode(code: TCode): T
where T : Codified<TCode>, T : Enum<T>, TCode : Serializable
= Codified.Enums.decode(java, code)
fun <T, TCode> KClass<T>.tryDecode(code: TCode): T?
where T : Codified<TCode>, T : Enum<T>, TCode : Serializable
= Codified.Enums.tryDecode(java, code)
Stavo per raccomandare lo stesso. Inoltre, vorrei rendere 'FromInt' return non-null come' Enum.valueOf (String) ':' map [type]?: Throw IllegalArgumentException() ' – mfulton26
Dato il supporto kotlin per null-safety, restituendo null dal metodo non mi infastidirebbe come in Java: il compilatore dovrà forzare il chiamante a gestire un valore restituito nullo e decidere cosa fare (lanciare o fare qualcos'altro). –
Buon punto. Grazie. – mfulton26