Откуда возникает метод Enum.valueOf(String)?

В Java SE 7 (и, скорее всего, в предыдущих версиях) класс Enum объявляется следующим образом:

 public abstract class Enum<E extends Enum<E>>
 extends Object
 implements Comparable<E>, Serializable

Класс Enum имеет статический метод с этой сигнатурой:

  T static<T extends Enum<T>> valueOf(Class<T> enumType, String name) 

Но нет статического метода: valueOf(String), определенного в классе Enum, и не вверх в иерархии, к которой принадлежит Enum.

Вопрос в том, откуда берется valueOf(String)? Является ли это признаком языка, то есть функцией, встроенной в компилятор?

Ответ 1

Этот метод неявно определяется компилятором.

Из документации:

Обратите внимание, что для определенного типа перечисления T вместо этого метода может использоваться неявно объявленный открытый статический метод T valueOf (String) для этого перечисления, чтобы сопоставить имя с соответствующей константой перечисления. Все константы типа enum могут быть получены путем вызова неявного публичного статического метода T [] values ​​() этого типа.

Из Спецификация языка Java, раздел 8.9.2:

Кроме того, если E является именем типа перечисления, то этот тип имеет следующие неявно объявленные статические методы:

/**
* Returns an array containing the constants of this enum 
* type, in the order they're declared.  This method may be
* used to iterate over the constants as follows:
*
*    for(E c : E.values())
*        System.out.println(c);
*
* @return an array containing the constants of this enum 
* type, in the order they're declared
*/
public static E[] values();

/**
* Returns the enum constant of this type with the specified
* name.
* The string must match exactly an identifier used to declare
* an enum constant in this type.  (Extraneous whitespace 
* characters are not permitted.)
* 
* @return the enum constant with the specified name
* @throws IllegalArgumentException if this enum type has no
* constant with the specified name
*/
public static E valueOf(String name);

Ответ 2

Я думаю, что это должна быть особенность языка. Для одного, чтобы создать перечисление, сделав одно, и ему не нужно расширять Enum:

public enum myEnum { red, blue, green }

Это само по себе является языковой функцией или вам нужно будет это сделать:

public class  MyEnum extends Enum { ..... }

Во-вторых, метод Enum.valueOf(Class<T> enumType, String name) должен быть создан компилятором, когда вы используете myEnum.valueOf(String name).

Это то, что казалось бы возможным, учитывая, что новый Enum является такой же языковой функцией, как и класс, который расширяется.