Реализация основана на API отражения Java.
Если вы определяете val для значений перечисления:
object Fruit extends Enumeration {
val Apple, Banana, Cherry = Value
}
Существуют методы для val в классе Fruit:
scala> Fruit.getClass.getMethods filter (_.getName.contains("Apple")) foreach println
public scala.Enumeration$Value line10$object$$iw$$iw$Fruit$.Apple()
toString
вызывает Enumeration.this.nameOf(i)
, если имя не задано явно. Этот метод попытается найти все методы в классе перечисления, возвращающие Value
экземпляров.
val methods = getClass.getMethods
for (m <- methods
if (classOf[Value].isAssignableFrom(m.getReturnType) &&
!java.lang.reflect.Modifier.isFinal(m.getModifiers) &&
m.getParameterTypes.isEmpty &&
isValDef(m)))
Это методы класса Fruit.
Затем для построения карты используются имя методов и идентификаторы значений перечисления id -> name
и извлекается имя из карты с идентификатором значения перечисления.
val name = m.getName
// invoke method to obtain actual `Value` instance
val value = m.invoke(this)
// invoke `id` method
val idMeth = classOf[Val].getMethod("id")
val id: Int = idMeth.invoke(value).asInstanceOf[java.lang.Integer].intValue()
Эта реализация может быть легко нарушена, если вы определите перечисление следующим образом:
object X extends Enumeration {
val Y = Value
}
object Fruit extends Enumeration {
val x = X.Y
val A,B,C = Value
}
Это Fruit.value возвращает object$Fruit.ValueSet(x, B, C)
, а не object$Fruit.ValueSet(A, B, C)
.