enum一个最不像class的class
enum一个最不像class的class
java枚举类型是jdk5出现的.它的出现主要为了解决一些有特殊意义,已经确定的,长度不会改变的集合.
//月份描述 public class Month { //月份名称 private final String name; //月份天数 private final int days; //构造子,给出一个月份名称,默认天数31天 private Month(String name) { this(name, 31); } //构造子,给出月份名称,月份天数 private Month(String name, int days) { this.name = name; this.days = days; } /* * 创建12个月份 * */ public static final Month JAN = new Month("January"); public static final Month FEB = new Month("February", 28); public static final Month MAR = new Month("March"); public static final Month APR = new Month("April", 30); public static final Month MAY = new Month("May"); public static final Month JUN = new Month("June", 30); public static final Month JUL = new Month("July"); public static final Month AUG = new Month("August"); public static final Month SEP = new Month("September", 30); public static final Month OCT = new Month("October"); public static final Month NOV = new Month("November", 30); public static final Month DEC = new Month("December"); //获取月份名称 public String getName() { return name; } //获取月份值 public int getDays() { return days; } }
public class Main { public static void main(String[] args) { System.out.println(Month.JAN);//demo7.Month@1b6d3586 System.out.println(Month.JAN.getName());//January System.out.println(Month.JAN.getDays());//31 } }
以上代码是创建了一个不可改变的月份表.内部实现就是一个在一个类中创建了12个实例对象.在客户端无需创建,直接获取没有月份的相关属性.以上写法在没有enum类型出现前,是最实用的写法.我们也可以看出
在enum类中,只提供查询方法,不提供修改方法.
在jdk1.5以后,创建enum类型就简单的多了.
public enum EnumMonth { JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV, DEC; }
public class Main { public static void main(String[] args) { System.out.println(EnumMonth.APR);//APR } }
这样声明一个只有12个月份的类是不是简单的多.而且我们可以看出.在客户端我们直接打印enum项打印的值,而不是地址值.那么这里边到底是怎么实现的呢?我们来看看反编译后的enum
public final class EnumMonth extends Enum { public static final EnumMonth JAN; public static final EnumMonth FEB; public static final EnumMonth MAR; public static final EnumMonth APR; public static final EnumMonth MAY; public static final EnumMonth JUN; public static final EnumMonth JUL; public static final EnumMonth AUG; public static final EnumMonth SEP; public static final EnumMonth OCT; public static final EnumMonth NOV; public static final EnumMonth DEC; private static final EnumMonth $VALUES[]; public static EnumMonth[] values() { return (EnumMonth[])$VALUES.clone(); } public static EnumMonth valueOf(String name) { return (EnumMonth)Enum.valueOf(demo7/EnumMonth, name); } private EnumMonth(String s, int i) { super(s, i); } static { JAN = new EnumMonth("JAN", 0); FEB = new EnumMonth("FEB", 1); MAR = new EnumMonth("MAR", 2); APR = new EnumMonth("APR", 3); MAY = new EnumMonth("MAY", 4); JUN = new EnumMonth("JUN", 5); JUL = new EnumMonth("JUL", 6); AUG = new EnumMonth("AUG", 7); SEP = new EnumMonth("SEP", 8); OCT = new EnumMonth("OCT", 9); NOV = new EnumMonth("NOV", 10); DEC = new EnumMonth("DEC", 11); $VALUES = (new EnumMonth[] { JAN, FEB, MAR, APR, MAY, JUN, JUL, AUG, SEP, OCT, NOV, DEC }); } }
从这个class文件我们能看出
EnumMonth继承自Enum.
EnumMonth有一个私有的两个参数的构造函数.
EnumMonth类中的枚举项也是自己通过实例化的构成的.
接下来我们来看看他的两个方法 valueOf() 和values()
public class Main { public static void main(String[] args) { System.out.println(EnumMonth.APR);//APR EnumMonth apr = EnumMonth.valueOf("APR"); System.out.println(apr);//APR EnumMonth[] values = EnumMonth.values();//JAN,FEB,MAR,APR,MAY,JUN,JUL,AUG,SEP,OCT,NOV,DEC, for (EnumMonth enumMonth : values){ System.out.print(enumMonth+","); } } }
这不是我们的重点,现在我们要找它的toString()方法,可以让枚举项直接打印出值的,肯定只有toString().前往父类Enum寻找
所以现在可以得出结论.当我们创建一个枚举类时,实际上是有一个两个参数构造函数.默认传递的实参是我们的枚举项字符串和索引.
那么,如果我们覆盖双参数的构造函数呢?
public enum DoubleEnumMonth { JAN("JAN",0), FEB("FEB",1), MAR("MAR",2); private DoubleEnumMonth(String a, int b) { } }
public class Main { public static void main(String[] args) { System.out.println(DoubleEnumMonth.JAN);//JAN } }
可以看出,我们虽然只声明了双参数的构造子,并没有声明属性,也没有this赋值,但是依然可以通过toString()的方式直接访问到属性值.那肯定是编译器帮我们调用了super(a,b);来看一下
public final class DoubleEnumMonth extends Enum { public static final DoubleEnumMonth JAN; public static final DoubleEnumMonth FEB; public static final DoubleEnumMonth MAR; private static final DoubleEnumMonth $VALUES[]; public static DoubleEnumMonth[] values() { return (DoubleEnumMonth[])$VALUES.clone(); } public static DoubleEnumMonth valueOf(String name) { return (DoubleEnumMonth)Enum.valueOf(demo7/DoubleEnumMonth, name); } private DoubleEnumMonth(String s, int i, String a, int b) { super(s, i); } static { JAN = new DoubleEnumMonth("JAN", 0, "JAN", 0); FEB = new DoubleEnumMonth("FEB", 1, "FEB", 1); MAR = new DoubleEnumMonth("MAR", 2, "MAR", 2); $VALUES = (new DoubleEnumMonth[] { JAN, FEB, MAR }); } }
这个结果,既有些意外之中,又意料之外.编译器,帮我们做了一个4个参数的构造函数.不用说前两个肯定还是name和index.至于后两个才是我们自己赋的值.
最后总结下.java的enum类型,自1.5出现后,就是为了简化我们创建一个集合类的不可改变类的.