枚举(enum):一组常量的集合,属于一种特殊的类,里面只包含一组有限的特定的对象。
● 枚举的两种实现方式:
● 自定义枚举应用案例:
要求创建季节(Seanson)对象,给出属性、构造器等结构,再添加枚举去实现调用并输出结果。
public class Enumeration01{public static void main(String[] args){System.out.println(Season.AUTUMN);System.out.println(Season.SPRING);}
}class Season{ //类private String name;private String desc; //描述public static final Seanson SPRING = new Season("春天","温暖");public static final Seanson WINTER = new Season("冬天","寒冷");public static final Seanson AUTUMN = new Seanson("秋天","凉爽");public static final Seanson SUMMER = new Seanson("夏天","炎热");//1.将构造器私有化,目的是防止直接创建对象//2.去掉setXxx方法,防止属性被修改//3.在Season内部,直接创建固定的对象//4.优化:加入final修饰符private Season(String name,String desc){this.name = name;this.desc = desc;}public String getName(){return name;}public String getDesc(){return desc;}public String toString(){return "Seanson{"+"name=" + name + '\'' +",desc=" + desc + '\'' + '}';}
}
运行效果

● 自定义类实现枚举小结
使用enum来实现前面的枚举案例
如果使用了enum来实现枚举类
- 使用关键字 enum 替代 class
- public static final Season SPRING = new Season(“春天”,“温暖”)可以替换为:
SPRING(“春天”,“温暖”) ==> 常量名(参数列表)- 如果有多个常量(对象),使用 " , " 号间隔即可
- 如果使用 enum 来实现枚举,要求将定义常量对象,写在最前面。
public class Enumeration01{public static void main(String[] args){System.out.println(Season.AUTUMN);System.out.println(Season.SPRING);}
}enum Season{ //类//public static final Season SPRING = new Season("春天","温暖")可以替换为://SPRING("春天","温暖") ==> 常量名(参数列表)SPRING("春天","温暖"),WINTER("冬天","寒冷"),AUTUMN("秋天","凉爽"),SUMMER("夏天","炎热");private String name;private String desc; //描述private Season(String name,String desc){this.name = name;this.desc = desc;}public String getName(){return name;}public String getDesc(){return desc;}public String toString(){return "Seanson{"+"name=" + name + '\'' +",desc=" + desc + '\'' + '}';}
}
运行效果

SUMMER("夏天","炎热"),Season;
enum Season{spring,summer;
}
enum Season{spring,summer;构造器...
}
enum Gender{BOY,GIRL;
}
答案:
上面的语法是正确的;有一个枚举类:Gender,没有属性;有两个枚举对象 BOY,GIRL,使用的是无参构造器创建。
如果修改一下上面的代码,添加一个有参构造器的话,结果又会怎么样?
enum Gender{BOY,GIRL;private Gender(String name){}
}
答案:
因为添加了构造器,程序检测时就会把原本默认的构造器给覆盖(把原本默认的无参构造器给覆盖了),变为有参构造器,所以添加了构造器的Gender类需要加上无参构造器。
enum Gender{BOY,GIRL;private Gender(String name){}private Gender(){}
}
在使用关键字enum时,会隐式继承Enum类,这样我们就可以使用Enum类相关的方法。
常用的enum方法有:
| 方法名 | 详细描述 |
|---|---|
| valueOf | 传递枚举类型的Class对象和枚举常量名称给静态方法valueOf,会得到与参数匹配的枚举常量 |
| toString | 得到当前枚举常量的名称。一般可以通过重写这个方使得到的结果更易读。 |
| equals | 在枚举类型中可以直接使用 " == " 来比较两个枚举常量是否相等。 |
| hashCode | Enum实现了hashCode()来和equals()保持一致。它也是不可变的。 |
| getDeclaringClass | 得到枚举常量所属枚举类型的Class对象。可以用它来判断两个枚举常量是否属于同一个枚举类型。 |
| name | 得到当前枚举常量的名称。建议先试用toString()。 |
| ordinal | 得到当前枚举常量的次序 |
| compareTo | 枚举类型实现了Comparable接口,这样可以比较两个枚举常量的大小(按照声明的顺序排列) |
| clone | 枚举类型不能被Clone,为了防止子类实现克隆方法,Enum实现了一个仅抛出CloneNotSupporttedException异常的不变Clone()。 |
1. name:得到当前枚举常量的名称。建议先试用toString()。
//使用Season2 枚举类,来演示各种方法
Season2 autumn = Season2.AUTUMN;
//输出枚举对象的名字
System.out.println(autumn.name());
运行效果

2. ordinal:输出的是该枚举对象的次序/编号,从0开始编号
//ordinal() 输出的是该枚举对象的次序/编号,从0开始编号
//AUTUMN 枚举对象是第三个 , 因此输出2
System.out.println(autumn.ordinal());
运行效果

这里补充一个增强for
//这里补充一个增强forint[] nums = {1,2,9};//普通的for循环System.out.println("=====普通的for=====");for(int i = 0; i < nums.length; i++){}System.out.println("=====增强的for=====");
//执行流程是:依次从nums数组中取出数据,赋给i,如果取出完毕,则退出forfor(int i : nums){System.out.println("i=" + i);}
}
运行效果

3. values :从反编译可以看出values方法,返回Season2[]含有定义的所有枚举对象
注意:Java编译器会自动在enum类型中插入values方法,Enum类里面没有
//从反编译可以看出values方法,返回Season2[]含有定义的所有枚举对象
Season2[] values = Season2.values();
System.out.println("===遍历输出枚举对象(增强for)====");
for(Season2 season: values){//增强for循环System.out.println(season);
}
运行效果

4. values:返回当前枚举类中所有的常量
//从反编译可以看出,values方法,返回Season2[]//含有定义的所有枚举对象Season2[] values = Season2.values();System.out.println("===遍历取出枚举对象(增强for)====");for(Season2 season : values){System.out.println(season);}
运行效果

5. valuesOf:将字符串转换成枚举对象,要求字符串必须为已有的常量名,否则报异常!
valuesOf的执行流程:
Season2 autumn1 = Season2.valuesOf("AUTUMN");
system.out.println("autumn1=" + autumn1);
system.out.println(autumn == autumn1);
运行结果

6. compareTo:比较两个枚举常量,比较的就是位置号(编号)!
compareTo运行流程:
就是把一个枚举对象与另一个枚举对象的编号进行比较
//这里的compareTo就是把Season2.AUTUMN枚举对象的编号和Season2.SUMMER枚举对象的编号进行比较
//而此时的Season2.AUTUMN的编号为:[2] - Seaso.SUMMER的编号:[3],所以得到结果为:-1
System.out.println(Season2.AUTUMN.compareTo(Season2.SUMMER));
运行效果

enum 类名 implements 接口1,接口2{}
enum实现的枚举类,仍然是一个类,所以还是可以实现接口的。
interface IPlaying{public void playing();
}
enum Music implements IPlaying{CLASSICMUISC;public void playing(){System.out.println("播放音乐...");}
}
题目如下:

代码实现:
public static void main(String[] args){//获取到所有的枚举对象,即数组Week[] weeks = Week.values();//遍历,使用增强forSystem.out.println("===所有星期的信息如下===");for(Week week : weeks){System.out.println(vweek + '/n');}
}Week class{//定义Week的枚举对象MONDAY("星期一"),TUESDAY("星期二"),WEDNESDAY("星期三"),THURSDAY("星期四"),FRIDAY("星期五"),SATURDAY("星期六"),SUNDAY("星期日");private String name;//构造器private Week(String name){this.name = name;}//重写toStringpublic String toString(){return name;}
}
运行效果

● 注解的理解
● Annotation的使用
使用Annotation时要在其前面增加@符号,并把该Annotation当成一个修饰符使用。用于修饰它支持的程序元素
● 三个基本的Annotation:
@Override:限定某个方法,是重写父类方法,该注解只能用于方法
Override使用演示
class Father{ //父类public void fly(){System.out.println("Father fly...");}
}class Son extends Father{ //子类//1.Overrdie注解,放在fly方法上,表示子类的fly方法时重写了父类的fly//2.这里如果没有写@Override,还是重写了父类fly//3.如果你写了@Override注解,编译器就会去检查该方法是否真的重写了//父类的方法,如果的确重写了,则编译通过,如果没有构成重写,则编译错误//4.@Override的定义//如果发小@interface,则表示一个注解类/*@Target(ElementType.METHOD)@Retention(RetentionPolicy.SOURCE)public @interface Override{}*/@Override //说明public void fly(){System.out.println("Son fly....");}
}
@Deprecated:用于表示某个程序元素(类、方法等)已过时
@Deprecated的说明:
public class Deprecated01{public static void main(String[] args){A a = new A();a.hi();System.out.println();}
}@Deprecated
class A {public int n1 = 10;public void hi(){}
}
如果在n1和hi的上面添加上Deprecated,则在你使用n1和hi的时候,会有中划线。
修改后:
@Deprecated
class A {@Deprecatedpublic int n1 = 10;@Deprecatedpublic void hi(){}
}
主方法调用时的中划线:

@SuppressWarnings:抑制编译器警告
当我们不希望看到系统给的黄色警告时,我们可以使用@SuppressWarnings注解来抑制警告信息
● SuppressWarnings作用范围
关于SuppressWarnings作用范围是和你放置的位置相关,比如:
@SuppressWarnings放置在main方法,那么抑制警告的范围就是main
通常我们可以放置具体的语句,比如:方法、类。
@SuppressWarnings使用演示
在使用Idea工具时,往往会出现很多黄色的警告,比如下面的代码,在写完时Idea右上角就会出现黄色的三角警告:
public class SuppressWarnings01{public static void main(String[] args){List list = new ArrayList();list.add("");list.add("");list.add("");int i;System.out.println(list.get(1));}
}
三角警告

如果我们不希望看到这些警告的时候,就可以使用@SuppressWarnings注解来抑制这些警告:
//在{""}中,可以写入你希望抑制的(不显示)警告信息
public class SuppressWarnings01{@SuppressWarnings({"rawtypes"})public static void main(String[] args){List list = new ArrayList();list.add("");list.add("");list.add("");int i;System.out.println(list.get(1));}
}
这时候就可以看见右上角的警告变少了

再加上其他的类型,就可以屏蔽掉所有的警告了。
@SuppressWarnings({"rawtypes","unchecked","unused"})
● SuppressWarnings能指定的警告类型有:
| all | 抑制所有警告 |
|---|---|
| boxing | 抑制与封装/拆装作业相关的警告 |
| cast | 抑制与强制转换型作业相关的警告 |
| dep-ann | 抑制与淘汰注释相关的警告 |
| deprecation | 抑制与淘汰的相关警告 |
| fallthrough | 抑制与switch陈述式中遗漏break相关的警告 |
| finally | 抑制与未传回finally区块相关的警告 |
| hiding | 抑制与隐藏变数的取余变数相关的警告 |
| incomplete-switch | 抑制与switch陈述式(enum case)中遗漏项目相关的警告 |
| javadoc | 抑制与javadoc相关的警告 |
| nls | 抑制与非nls子串文字相关的警告 |
| null | 抑制与空值分析相关的警告 |
| rawtypes | 抑制与eaw类型相关的警告 |
| resource | 抑制与使用Closeable类型相关的资源相关的警告 |
| restnicthon | 抑制与使用不建议或进制参照相关的警告 |
| serial | 抑制与可序列化的类别遗漏serialVresionUID栏位相关的警告 |
| static-access | 抑制与静态存取不正确相关的警告 |
| static-method | 抑制与可能警告为static的方法相关的警告 |
| super | 抑制与置换方法相关但不含super呼叫的警告 |
| synthetic-access | 抑制与内部类别的存取未最佳化相关的警告 |
| sync-override | 抑制因为置换同步方法面遗漏同步化的警告 |
| unchecked | 抑制与未检查的作业相关的警告 |
| unqualified-field-access | 抑制与栏位存取不合格相关的警告 |
| unused | 抑制与未用的程式码及停用的程式码相关的警告 |
● @SuppressWarnings注解的各种值
查看@Target
@Target({TYPE,FIELD,METHOD,PARAMETER,CONSTRUCTOR,LOCAL_VARIABLE})@Retention(RetentionPolicy.SOURCE)public @interface SuppressWarnings{String[] value();
}
(1) 可修饰的值就为大括号内的值:TYPE,FIELD,METHOD,PARAMETER,CONSTRUCTOR,LOCAL_VARIABLE
(2)该注解类有数组 String[] values() 设置一个数组,比如{“rawtypes”,“unchecked”,“unused”}
● 元注解的基本介绍
JDK 的元 Annotaion 用于修饰其他 Annotation
元注解:本身作用不大,主要是看源码时知道是个什么东西。
● 元注解的种类(使用不多,了解,不用深入)
1)Retention //指定注解的作用范围,三种SOURCE,CLASS,RUNTIME
2)Target //指定注解可以在哪些地方使用
3)Documented //指定该注解是否会在javadoc体现
4)Inherited //子类会继承父类注解
@Retention:用于指定该Annotation可以保留多长时间,只能用于修饰一个Annotation定义。
@Retention包含一个RetentionPolicy类型的成员变量,使用@Retention时必须为该value成员变量指定值
● @Retention的三种值
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.SOURCE)
public @interface Override{
}
Target:用于修饰Annotation定义。
用于修饰Annotation定义,用于指定被修饰的Annotation能用于修饰程序的元素为:
@Target({TYPE,FIELD,METHOD,PARAMETER,CONSTRUCTOR,LOCAL_VARIABLE,PACKAGE,PARAMETER})
@Target也包含一个名为value的成员变量。
@Documented:用于指定被该元Annotation修饰的Annotation类将被javadoc工具提取成文档,即在生成文档时,可以看到该注解。
注意:
定义为Documented的注解必须设置Retention值为RUNTIME。
Inherited:被它修饰的Annotation将具有继承性,如果某个类使用了被@Inherited修饰的Annotation,则其子类将自动具有该注解
注意:
实际应用中,使用较少,了解即可。
下一篇:SSRF漏洞 | iwebsec