Java高级-枚举类与注解

10.1.枚举类的使用: 入门
  • 类的对象只有有限个,确定的

  • 自定义枚举类

一.枚举类的使用
1.枚举类的理解: 类的对象只有有限个,确定的.称之为枚举类
2.当要定义一组常量时,强烈建议用枚举类
3.如果枚举类中只有一个对象,则可以作为单例模式的实现方式

// 自定义枚举类的展现
public class SeasonTest {
    public static void main(String[] args) {
        // 调用枚举类的对象
        Season1 spring = Season1.SPRING;
        System.out.println(spring);

    }
}
// 自定义枚举类,还是继承于Object类
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;
    }
    // 5.诉求2.提供重写的toString()

    @Override
    public String toString() {
        return "Season{" +
                "seasonName='" + seasonName + '\'' +
                ", seasonDesc='" + seasonDesc + '\'' +
                '}';
    }
}
  • 使用enum关键字定义枚举类
  • Enum类的常用方法

说明: 定义的枚举类默认继承于java.lang.Enum类

public class SeasonTest1 {
    public static void main(String[] args) {
        Season1 summer = Season1.SUMMER;
        System.out.println(summer);
        // season1.class表示类的对象
        System.out.println(Season1.class.getSuperclass());// java.lang.Enum
        // toString(): 获取枚举类对象的名字/常量名
        System.out.println(summer.toString());
        // values(): 生成一个枚举类对象常量的数组,可以看枚举类状态
        Season1[] values = Season1.values();
        for (int i = 0; i < values.length; i++) {
            System.out.println(values[i]);
        }
        Thread.State[] values1 = Thread.State.values();
        for (int i = 0; i < values1.length; i++) {
            System.out.println(values1[i]);
        }
        // valueOf(String objName): 根据对象名传入的对象名,判断是否存是objName对象;找指定名的对象
        // 如果没有objName的枚举类对象,则抛异常: IllegalArgumentException
        Season1 winter = Season1.valueOf("winter");
        System.out.println(winter);
    }
}
// 使用enum关键字枚举类
enum Season1 {
    // 1.提供(创建)当前枚举类的对象,多个对象之间用","隔开,末尾对象用";"结束
    SPRING("春天", "春暖花开"),// 括号中含属性
    SUMMER("夏天", "夏日炎炎"),
    AUTUMN("秋天", "秋高气爽"),
    WINTER("冬天", "冰天雪地");
    //2.声明Season对象的属性:private final修饰,对象地址不变
    private final String seasonName;
    // 2.私有化构造器,并给对象属性赋值,在构造器里赋值每个对象属性值可以不一样
    private final String seasonDesc;

    private Season1(String seasonName, String seasonDesc){
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }
    // 4.诉求1.获取枚举类对象的属性

    public String getSeasonName() {
        return seasonName;
    }

    public String getSeasonDesc() {
        return seasonDesc;
    }

    // 5.诉求2.提供重写的toString()
    // 已经给重写过了,默认打印对象的名字
    /*@Override
    public String toString() {
        return "Season1{" +
                "seasonName='" + seasonName + '\'' +
                ", seasonDesc='" + seasonDesc + '\'' +
                '}';
    }*/
}
  • 使用enum关键字定义的枚举类实现接口

情况一: 实现接口,在enum类中实现抽象方法
情况二: 让枚举类的对象分别实现接口中的抽象方法,体现每个枚举类对象的方法体都不一样了

public class SeasonTest1 {
    public static void main(String[] args) {
        Season1 summer = Season1.SUMMER;
        System.out.println(summer);
        // season1.class表示类的对象
        System.out.println(Season1.class.getSuperclass());// java.lang.Enum
        // toString(): 获取枚举类对象的名字/常量名
        System.out.println(summer.toString());
        // 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对象;找指定名的对象
        // 如果没有objName的枚举类对象,则抛异常: IllegalArgumentException
        Season1 winter = Season1.valueOf("WINTER");
        System.out.println(winter);
        winter.show();// 调用实现接口的方法
    }
}
// 定义一个接口
interface Info{
    void show();// 省略 public abstract
}
// 使用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("2002年的第一场雪");
        }
    };
    //2.声明Season对象的属性:private final修饰,对象地址不变
    private final String seasonName;
    // 2.私有化构造器,并给对象属性赋值,在构造器里赋值每个对象属性值可以不一样
    private final String seasonDesc;

    private Season1(String seasonName, String seasonDesc){
        this.seasonName = seasonName;
        this.seasonDesc = seasonDesc;
    }
    // 4.诉求1.获取枚举类对象的属性

    public String getSeasonName() {
        return seasonName;
    }

    public String getSeasonDesc() {
        return seasonDesc;
    }

    // 5.诉求2.提供重写的toString()
    // 已经给重写过了,默认打印对象的名字
    /*@Override
    public String toString() {
        return "Season1{" +
                "seasonName='" + seasonName + '\'' +
                ", seasonDesc='" + seasonDesc + '\'' +
                '}';
    }*/

    @Override
    public void show() {
        System.out.println("这是个季节");
    }
}
10.2.注解(Annotation)
  • 如何自定义注解

① 注解声明为: @interface
② 内部定义成员,通常使用value表示
③ 可以指定成员的默认值,用default定义
④ 如果自定义注解没有成员,表明是一个标识作用,例如Override
如果注解有成员,在用注解时,要指明成员的值
自定义注解必须配上注解的信息处理流程(使用反射)才有意义

@MyAnnotation(value = "hi") // 有成员需要指定值,注解加默认值,小括号可省略
public @interface MyAnnotation {
    // 自定义注解
    String[] value() default "hello"; // 可加默认值,
}

4.jdk提供的4种元注解
元注解: 对现有的注解进行解释说明的注解
Retention: 指定所修饰的Annotation 的生命周期: SOURCE\CLASS(默认行为)\RUNTIME(只有声明为RUNYIME生命周期的注解,才能通过反射获取,必须加载到内存中
Target: 指明修饰的注解可以修饰其他哪些结构
出现频率较低
Documented: 表示所修饰的注解在被javadoc解析时,保留下来,默认的时候注解是不保留的
Inherited:被它修饰的Annotation将具有继承性.父类有了被它修饰的注解,子类也会有
5.通过反射获取注解信息 — 到反射内容是系统讲解
6.jdk8中注解的新特性: 可重复注解,类型注解
6.1.可重复注解: 可以重复定义多个的注解
①在MyAnnotation上声明@Repeatable,成员值为MyAnnotation.class ; 就让这两个注解关联在一起
MyAnnotation的Target和Rentention等其他元注解和MyAnnotation相同
6.2.类型注解:
ElementType.TYPE_PARAMETER: 表示该注解能写在类型变量的声明语句中(如:泛型声明)
ElementType.TYPE_USE: 表示该注解能写在使用类型的任何语句中

public class AnnotationTest {
    public static void main(String[] args) {
        Person p = new Student();
        p.walk();
        @Deprecated // 表示修饰的元素(类,方法等)已过时
        Date date = new Date(2020, 10, 11);
        System.out.println(date);
        @SuppressWarnings("unused")// 抑制编译器编译时警告
        int num = 10;
        @SuppressWarnings({"unused","rawtypes"})
        ArrayList list = new ArrayList();
    }
    @Test
    public void testGetAnnotation() {
        Class clazz = Student.class;
        Annotation[] annotations = clazz.getAnnotations();
        for (int i = 0; i < annotations.length; i++) {
            System.out.println(annotations[i]);
        }
    }
}
// 重复注解写法,但这样写会报错
 @MyAnnotation(value = "hi") // 有成员需要指定值,注解加默认值,小括号可省略
 @MyAnnotation(value = "abc")
// 以前的写法要重新创建一个注解数组,呈现出来还是一个结构
// @MyAnnotations({@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;
    }

    public void eat(){
        System.out.println("人吃饭");
    }
    public void walk(){
        System.out.println("人走路");
    }
}
interface Info{
    void show();
}
class Student extends Person implements Info{
    @Override
    public void walk() {
        System.out.println("学生走路");
    }
    @Override
    public void show() {

    }
}
// 在注解中的Target注解加入TYPE_PARAMETER,TYPE_USE就不会报错
class Generic<@MyAnnotation T>{
    public void show()throws @MyAnnotation RuntimeException{
        ArrayList<@MyAnnotation String> list = new ArrayList<>();
        int num = (@MyAnnotation int) 10L;
    }
}
@Inherited
@Retention(RetentionPolicy.RUNTIME)// 可以通过反射获取此注解
@Repeatable(MyAnnotations.class)
@Target({ElementType.TYPE, ElementType.FIELD,ElementType.METHOD,ElementType.PARAMETER,ElementType.CONSTRUCTOR,ElementType.LOCAL_VARIABLE,ElementType.TYPE_PARAMETER,ElementType.TYPE_USE})
public @interface MyAnnotation {
    // 自定义注解
    String[] value() default "hello"; // 可加默认值,
}
@Inherited
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.FIELD,ElementType.METHOD,ElementType.PARAMETER,ElementType.CONSTRUCTOR,ElementType.LOCAL_VARIABLE})
public @interface MyAnnotations {
    // 声明一个MyAnnotation类型的数组
    MyAnnotation[] value(); // 这是个属性
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值