Java枚举类和注解

1、枚举类

  • 当需要定义一组常量时,强烈建议使用枚举类。
  • 枚举类的实现
    • JDK1.5之前需要自定义枚举类。
    • JDK 1.5 新增的 enum 关键字用于定义枚举类。
  • 若枚举只有一个对象, 则可以作为一种单例模式的实现方式。
  • 枚举类的属性
    • 枚举类对象的属性不应允许被改动, 所以应该使用 private final 修饰。
    • 枚举类的使用 private final 修饰的属性应该在构造器中为其赋值。
    • 若枚举类显式的定义了带参数的构造器, 则在列出枚举值时也必须对应的传入参数。

1.1 自定义枚举类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
public class SeasonTest {
public static void main(String[] args) {
Season spring = Season.SPRING;
System.out.println(spring);//Season{seasonName='春天', seasonDesc='春暖花开'}
}
}
//自定义枚举类
class Season{
//1.声明Season对象的属性:private final修饰
private final String seasonName;
private final String seasonDesc;

//2.私有化类的构造器,并给对象属性赋值
private Season(String seasonName,String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}

//3.提供当前枚举类的多个对象:public static final的
public static final Season SPRING = new Season("春天","春暖花开");
public static final Season SUMMER = new Season("夏天","夏日炎炎");
public static final Season AUTUMN = new Season("秋天","秋高气爽");
public static final Season WINTER = new Season("冬天","冰天雪地");

//4.其他诉求1:获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}

public String getSeasonDesc() {
return seasonDesc;
}

//4.其他诉求1:提供toString()
@Override
public String toString() {
return "Season{" +
"seasonName='" + seasonName + '\'' +
", seasonDesc='" + seasonDesc + '\'' +
'}';
}
}

1.2 enum定义枚举类

  • 使用 enum 定义的枚举类默认继承了 java.lang.Enum类,因此不能再继承其他类。
  • 枚举类的构造器只能使用 private 权限修饰符。
  • 枚举类的所有实例必须在枚举类中显式列出(, 分隔 ; 结尾)。列出的实例系统会自动添加 public static final 修饰。
  • 必须在枚举类的第一行声明枚举类对象。
  • Enum类的主要方法:
  • 和普通 Java 类一样,枚举类可以实现一个或多个。
  • 若每个枚举值在调用实现的接口方法呈现相同的行为方式,则只要统一实现该方法即可。
  • 若需要每个枚举值在调用实现的接口方法呈现出不同的行为方式, 则可以让每个枚举值分别来实现该方法。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
public class SeasonTest {
public static void main(String[] args) {
Season1 summer = Season1.SUMMER;
//toString():返回枚举类对象的名称
System.out.println(summer.toString());//SUMMER
System.out.println(Season1.class.getSuperclass());//class java.lang.Enum

//values():返回所有的枚举类对象构成的数组
Season1[] values = Season1.values();
for(int i = 0;i < values.length;i++){
System.out.println(values[i]);
values[i].show();
}

Thread.State[] values1 = Thread.State.values();
for (int i = 0; i < values1.length; i++) {
System.out.println(values1[i]);
}
//valueOf(String objName):返回枚举类中对象名是objName的对象。
Season1 winter = Season1.valueOf("WINTER");
//如果没有objName的枚举类对象,则抛异常:IllegalArgumentException
//Season1 winter = Season1.valueOf("WINTER1");
System.out.println(winter);//WINTER
winter.show();//大约在冬季
}
}

interface Info{
void show();
}

//使用enum关键字枚举类
//定义的枚举类默认继承于java.lang.Enum类
enum Season1 implements Info{
//1.提供当前枚举类的对象,多个对象之间用","隔开,末尾对象";"结束
SPRING("春天","春暖花开"){
@Override
public void show() {
System.out.println("春天在哪里?");
}
},
SUMMER("夏天","夏日炎炎"){
@Override
public void show() {
System.out.println("宁夏");
}
},
AUTUMN("秋天","秋高气爽"){
@Override
public void show() {
System.out.println("秋天不回来");
}
},
WINTER("冬天","冰天雪地"){
@Override
public void show() {
System.out.println("大约在冬季");
}
};

//2.声明Season对象的属性:private final修饰
private final String seasonName;
private final String seasonDesc;

//3.私有化类的构造器,并给对象属性赋值
private Season1(String seasonName,String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}

//4.其他诉求1:获取枚举类对象的属性
public String getSeasonName() {
return seasonName;
}

public String getSeasonDesc() {
return seasonDesc;
}

// //4.其他诉求1:提供toString()
// @Override
// public String toString() {
// return "Season1{" +
// "seasonName='" + seasonName + '\'' +
// ", seasonDesc='" + seasonDesc + '\'' +
// '}';
// }

// @Override
// public void show() {
// System.out.println("这是一个季节");
// }
}

2、注解

2.1 概述

  • 从 JDK 5.0 开始, Java 增加了对元数据(MetaData) 的支持, 也就是Annotation(注解)。
  • Annotation 可以像修饰符一样被使用, 可用于修饰包,类, 构造器, 方法, 成员变量, 参数, 局部变量的声明, 这些信息被保存在 Annotation 的 “name=value” 对中。
  • 使用 Annotation 时要在其前面增加 @ 符号, 并把该 Annotation 当成一个修饰符使用。用于修饰它支持的程序元素。
  • 未来的开发模式都是基于注解的,JPA是基于注解的,Spring2.5以上都是基于注解的,Hibernate3.x以后也是基于注解的,现在的Struts2有一部分也是基于注解的了,注解是一种趋势,一定程度上可以说:框架 = 注解 + 反射 + 设计模式

2.2 注解使用示例

  • 生成文档相关的注解

    • @author 标明开发该类模块的作者,多个作者之间使用,分割。
    • @version 标明该类模块的版本。
    • @see 参考转向,也就是相关主题。
    • @since 从哪个版本开始增加的。
    • @param 对方法中某参数的说明,如果没有参数就不能写。
    • @return 对方法返回值的说明,如果方法的返回值类型是void就不能写。
    • @exception 对方法可能抛出的异常进行说明 ,如果方法没有用throws显式抛出的异常就不能写其中。
  • 在编译时进行格式检查(JDK内置的三个基本注解)

    • @Override: 限定重写父类方法, 该注解只能用于方法。
    • @Deprecated: 用于表示所修饰的元素(类, 方法等)已过时。通常是因为所修饰的结构危险或存在更好的选择。
    • @SuppressWarnings: 抑制编译器警告。

2.3 自定义注解

  • Annotation 的成员变量在 Annotation 定义中以无参数方法的形式来声明。其方法名和返回值定义了该成员的名字和类型。我们称为配置参数。类型只能是八种基本数据类型、String类型、Class类型、enum类型、Annotation类型、以上所有类型的数组。

  • 如果只有一个参数成员,建议使用参数名为value。

  • 可以在定义 Annotation 的成员变量时为其指定初始值, 指定成员变量的初始值可使用 default 关键字。

  • 如果定义的注解含有配置参数,那么使用时必须指定参数值,除非它有默认值。格式是“参数名 = 参数值”
    ,如果只有一个参数成员,且名称为value,可以省略“value=”。

  • jdk 提供的4种元注解(对现有的注解进行解释说明的注解):

    • Retention:指定所修饰的 Annotation 的生命周期:SOURCE\CLASS(默认行为)\RUNTIME,只有声明为RUNTIME生命周期的注解,才能通过反射获取。
    • **Target:**用于指定被修饰的 Annotation 能用于修饰哪些程序元素。
    • **Documented:**表示所修饰的注解在被javadoc解析时,保留下来。
    • **Inherited:**被它修饰的 Annotation 将具有继承性。
  • 步骤

    • ① 注解声明为:@interface。
    • ② 内部定义成员,通常使用value表示。
    • ③ 可以指定成员的默认值,使用default定义。
    • ④ 如果自定义注解没有成员,表明是一个标识作用。
  • JDK1.8之后,关于元注解@Target的参数类型ElementType枚举值多了两个:TYPE_PARAMETER,TYPE_USE。

    • ElementType.TYPE_PARAMETER 表示该注解能写在类型变量的声明语句中(如:泛型声明)。
    • ElementType.TYPE_USE 表示该注解能写在使用类型的任何语句中。
  • Java 8对注解处理提供了两点改进:可重复的注解及可用于类型的注解。此外,反射也得到了加强,在Java8中能够得到方法参数的名称。这会简化标注在方法参数上的注解。

    • ① 在MyAnnotation上声明@Repeatable,成员值为MyAnnotations.class。
    • ② MyAnnotation的Target和Retention等元注解与MyAnnotations相同。
    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    38
    39
    40
    41
    42
    43
    44
    45
    46
    47
    48
    49
    50
    51
    52
    53
    54
    55
    56
    57
    58
    59
    60
    61
    62
    63
    @Inherited
    @Repeatable(MyAnnotations.class)
    @Retention(RetentionPolicy.RUNTIME)
    @Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE,TYPE_PARAMETER,TYPE_USE})
    public @interface MyAnnotation {
    String value() default "hello";
    }

    @Inherited
    @Retention(RetentionPolicy.RUNTIME)
    @Target({TYPE, FIELD, METHOD, PARAMETER, CONSTRUCTOR, LOCAL_VARIABLE})
    public @interface MyAnnotations {

    MyAnnotation[] value();
    }

    @MyAnnotation(value="hi")
    @MyAnnotation(value="abc")
    class Person{
    private String name;
    private int age;

    public Person() {
    }
    @MyAnnotation
    public Person(String name, int age) {
    this.name = name;
    this.age = age;
    }
    @MyAnnotation
    public void walk(){
    System.out.println("人走路");
    }
    public void eat(){
    System.out.println("人吃饭");
    }
    }

    interface Info{
    void show();
    }

    class Student extends Person implements Info{

    @Override
    public void walk() {
    System.out.println("学生走路");
    }

    public void show() {

    }
    }

    class Generic<@MyAnnotation T>{

    public void show() throws @MyAnnotation RuntimeException{

    ArrayList<@MyAnnotation String> list = new ArrayList<>();

    int num = (@MyAnnotation int) 10L;
    }
    }