第9章 枚举&注解

枚举类

枚举类入门

  • JDK1.5之前需要自定义枚举类。
  • JDK 1.5 新增的 enum关键字用于定义枚举类。
  • 若枚举只有一个成员, 则可以作为一种单例模式的实现方式。

枚举类的属性

  • 枚举类对象的属性不应允许被改动。所以,应该使用private final修饰。

枚举类的使用private final修饰。 修饰的属性应该在构造器中为其赋值。 若枚举类显式的定义了带参数的构造器, 则在列出枚举值时也必须对应的传入参数。

Enum枚举类

  • 必须在枚举类的第一行声明枚举类对象。
  • 枚举类和普通类的区别:

使用enum定义的枚举类默认继承了java.lang.Enum。 枚举类的构造器只能使用 private 访问控制符。 枚举类的所有实例必须在枚举类中显式列出(, 分隔; 结尾)。列出的实例系统会自动添加public static final修饰。

  • JDK 1.5中可以在switch表达式中使用Enum定义的枚举类的对象作为表达式,case子句可以直接使用枚举值的名字,无需添加枚举类作为限定。

  • 枚举类的主要方法:

values()方法:返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值valueOf(String str):可以把一个字符串转为对应的枚举类对象。要求字符串必须是枚举类对象的“名字”。如不是会有运行时异常。

使用Enum定义的 Season

public enum Season {

    SPRING("春天", "春风又绿江南岸"),
    SUMMER("夏天", "映日荷花别样红"),
    AUTUMN("秋天", "秋水共长天一色"),
    WINTER("冬天", "窗含西岭千秋雪");

    private final String name;
    private final String desc;

    Season(String name, String desc) {
        this.name = name;
        this.desc = desc;
    }

    public String getName() {
        return name;
    }

    public String getDesc() {
        return desc;
    }
}

枚举的方法

实现接口的枚举类

  • 和普通 Java 类一样,枚举类可以实现一个或多个接口。
  • 若需要每个枚举值在调用实现的接口方法呈现出不同的行为方式,则可以让每个枚举值分别来实现该方法。

注解Annotation

注解概述

  • 从 JDK 5.0 开始,Java 增加了对元数据(MetaData) 的支持, 也就是 Annotation(注解)。
  • Annotation 其实就是代码里的特殊标记,这些标记可以在编译,类加载,运行时被读取,并执行相应的处理。通过使用Annotation,程序员可以在不改变原有逻辑的情况下,在源文件中嵌入一些补充信息。
  • Annotation 可以像修饰符一样被使用,可用于修饰包,类,构造器, 方法,成员变量,参数,局部变量的声明,这些信息被保存在 Annotation 的 “name=value” 对中。
  • Annotation 能被用来为程序元素(类,方法,成员变量等) 设置元数据。

基本的 Annotation

  • 使用Annotation时要在其前面增加**@符号**,并把该Annotation当成一个修饰符使用。用于修饰它支持的程序元素。
  • 三个基本的 Annotation:

@Override:限定重写父类方法,该注释只能用于方法。 @Deprecated:用于表示某个程序元素(类, 方法等)已过时。 @SuppressWarnings:抑制编译器警告。

自定义 Annotation

  • 定义新的 Annotation 类型使用 @interface 关键字
  • Annotation 的成员变量在 Annotation 定义中以无参数方法的形式来声明。 其方法名和返回值定义了该成员的名字和类型。
  • 可以在定义 Annotation的成员变量时为其指定初始值,指定成员变量的初始值可使用 default 关键字
public @interface MyAnnotation {
    String name() default "baidu";
}
  • 没有成员定义的 Annotation 称为标记;包含成员变量的 Annotation 称为元数据 Annotation

提取 Annotation 信息

  • JDK 5.0 在 java.lang.reflect 包下新增了 AnnotatedElement接口,该接口代表程序中可以接受注解的程序元素。
  • 当一个Annotation类型被定义为运行时Annotation后,该注释才是运行时可见,当 class 文件被载入时保存在 class 文件中的 Annotation 才会被虚拟机读取。
  • 程序可以调用 AnnotationElement 对象的如下方法来访问 Annotation 信息。

JDK 的元 Annotation

  • JDK 的元 Annotation 用于修饰其他 Annotation 定义。
  • JDK5.0 提供了专门在注解上的注解类型,分别是:

@Retention @Target @Documented @Inherited

  • @Retention: 只能用于修饰一个 Annotation 定义,用于指定该 Annotation 可以保留多长时间,@Rentention 包含一个 RetentionPolicy 类型的成员变量,使用 @Rentention 时必须为该 value 成员变量指定值:

RetentionPolicy.SOURCE: 编译器直接丢弃这种策略的注释。 RetentionPolicy.CLASS: 编译器将把注释记录在 class 文件中. 当运行 Java 程序时, JVM 不会保留注解。 这是默认值。 RetentionPolicy.RUNTIME:编译器将把注释记录在 class 文件中. 当运行 Java 程序时, JVM 会保留注释。 程序可以通过反射获取该注释。

public enum RetentionPolicy{
	SOURCE,
	CLASS,
	RUNTIME
}
@Retention(RetentionPolicy.SOURCE)
@interface MyAnnotation1{}
@interface MyAnnotation2{}

@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation3{}
  • @Target: 用于修饰 Annotation 定义, 用于指定被修饰的 Annotation 能用于修饰哪些程序元素。@Target 也包含一个名为 value 的成员变量。
  • @Documented: 用于指定被该元 Annotation 修饰的 Annotation 类将被 javadoc 工具提取成文档。

定义为@Documented的注解必须设置@Retention值为RUNTIME

  • @Inherited:被它修饰的Annotation将具有继承性。如果某个类使用了被@Inherited修饰的Annotation,则其子类将自动具有该注解。

实际应用中,使用较少。

练 习

1.编写一个Person类,使用@Override注解它的toString()方法。

public class Person {
    private String name;
    public Person(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                '}';
    }
}

2.自定义一个名为“MyTiger”的注解类型,它只可以使用在方法上,带一个String类型的value属性,然后在第1题中的Person类上正确使用。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
@interface MyTiger {
    String value() default "baidu";
}

public class Person {
    private String name;
    public Person(String name) {
        this.name = name;
    }
    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                '}';
    }

    @MyTiger(value = "google")
    public void eat() {
        //通过反射,取得 @MyTiger注解的 value值
        try {
            MyTiger anno = Person.class.getMethod("eat").getAnnotation(MyTiger.class);
            System.out.println(anno.value());
        } catch (Exception e) {
            e.printStackTrace();
        }
        System.out.println("我想吃包子");
    }

    public static void main(String[] args) {
        Person p = new Person("张三");
        System.out.println(p);
        p.eat();
    }
}

转载于:https://my.oschina.net/mondayer/blog/3027131

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值