枚举、注解

枚举

类的对象只有有限个,确定的;当需要定义一组常量时,最好使用枚举类
1.如何定义枚举类
方式一:jdk5.0之前,自定义枚举类

public class Enum01 {
    public static void main(String[] args) {
        Season autumn = Season.Autumn;
        System.out.println(autumn);
    }
}
//自定义枚举
class Season{
    //final修饰属性三种方式:显示赋值,代码块赋值,构造器中赋值
    private final String name;//final定义的变量没有set方法,只有get方法
    private final String text;

    private Season(String name, String text) {
        this.name = name;
        this.text = text;
    }
    //自定义的枚举对象
    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("冬","白雪皑皑");

    public String getName() {
        return name;
    }

    public String getText() {
        return text;
    }

    @Override
    public String toString() {
        return "Season{" +
                "name='" + name + '\'' +
                ", text='" + text + '\'' +
                '}';
    }
}

方式二:jdk5.0可以使用enum关键字定义枚举类

public class Eunm02{
    public static void main(String[] args) {
        Season1 autumn = Season1.AUTUMN;
        System.out.println(autumn);
        //返回表示此 Class 所表示的实体(类、接口、基本类型或 void)的超类的 Class
        System.out.println(Season1.class.getSuperclass());//java.lang.Enum
    }
}

/*
*enum关键字枚举
* 1.定义的枚举对象必须在首行,不同对象之间使用,隔开,
* 2.默认重写了toString方法时,返回枚举常量的名称
* 3.定义的枚举类默认继承于java.lang.Enum包
*/
enum Season1{
    //必须定义在首行
    SPRING ("春","春意盎然"),
    SUMMER("夏","清爽透凉"),
    AUTUMN ("秋","落叶缤纷"),
    WINTER ("冬","白雪皑皑");
    //final修饰属性三种方式:显示赋值,代码块赋值,构造器中赋值
    private final String name;//final定义的变量没有set方法,只有get方法
    private final String text;
    //此构造方法用来定义枚举对象的格式
    private Season1(String name, String text) {
        this.name = name;
        this.text = text;
    }

    public String getName() {
        return name;
    }

    public String getText() {
        return text;
    }

//    @Override
//    public String toString() {
//        return "Season1{" +
//                "name='" + name + '\'' +
//                ", text='" + text + '\'' +
//                '}';
//    }
}

常用方法:

//values()返回枚举类型对象数组
        Season1[] values = Season1.values();
        for (int i = 0; i < values.length; i++) {
            System.out.println(values[i]);
        }
        //返回当前枚举对象常量名称
        System.out.println(autumn.toString());
        //valueOf("obj")返回带obj枚举类型的枚举常量
        Season1 autumn1 = Season1.valueOf("AUTUMN");
        System.out.println(autumn1);

使用enum关键字定义的枚举类实现接口
定义接口

interface  Test{
    void show();
}

不同对象重写同一个方法

//不同对象调用的方法得到不同的结果
enum Season2 implements Test{
    //必须定义在首行
    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("冬");
        }
    };
}

所有对象重写同一个方法

enum Season2 implements Test{
//不同对象调用的结果相同
@Override
    public void show() {
        System.out.println("四季");
    }
}

测试

//方式一:循环调用
 Season2[] values = Season2.values();
        for (int i = 0; i < values.length; i++) {
            System.out.println(values[i]);
            values[i].show();

        }
//方式二:直接调用
Season2.AUTUMN.show();
Season2.SPRING.show();

注解

自定义注解:
(1)注解声明为:@interface
(2)内部定义成员,通常使用value表示

public @interface MyAnnotation {
    String value();//可以设置自己需要的格式
}
//此时注解必须赋值
@MyAnnotation(value = "zhangsan")
public class AnnotationTest {

}

(3)可以指定成员的默认值,使用default定义

public @interface MyAnnotation {
    String value() default "ok";//给该注解默认赋值
}
//注解可以不赋值
@MyAnnotation()
public class AnnotationTest {

}

(4)若自定义注解没有成员,表明是一个标识作用
若注解有成员,在使用注解时,需要指明成员的值,同时自定义注解必须配上注解的信息处理流程(使用反射)才有意义

public @interface MyAnnotation {

}
@MyAnnotation
public class AnnotationTest {

}
jdk中的4种元注解

(元数据:String name = "xiaohong";其中Stirng name表示元数据)
元注解:对现有注解进行解释说明的注解
Retention:指定所修饰的Annotation的生命周期:SOURCE\CLASS(默认行为)\RUNTIME(只有声明为RUNTIME生命周期的注解,才能通过反射获取)
Target:用来指定被修饰的Annotation能用于修饰哪些程序元素:如:METHOD、
Documentend:表示所修饰的注解在被javadoc解析时,保留下来
Inherited:被他修饰的Annotation具有继承性;即父类声明的注解,子类也有

jdk8中注解的新特性:可重复注解、类型注解

可重复注解:
(1)在MyAnnotation 上声明@Repeatable(MyAnnotations.class);MyAnnotations.class表示成员值
(2)MyAnnotation的注解其余注解需要和MyAnnotations的相同

//MyAnnotations和MyAnnotation的注解类型信息必须一致
@Target(ElementType.TYPE_USE)
public @interface MyAnnotations {
    MyAnnotation[] value();//定义可重复注解时使用
}

@Repeatable(MyAnnotations.class)
@Target(ElementType.TYPE_USE)
public @interface MyAnnotation {
    String value();
}

//jdk8之前多个相同类型注解的方式
//@MyAnnotations({@MyAnnotation(value="hi"),@MyAnnotation(value="hello")})

//jdk8后不同注解信息
@MyAnnotation(value = "java")
@MyAnnotation(value = "python")
public class AnnotationTest{

}

类型注解:
ElementType.TYPE_PARAMETER:表示该注解能写在类型变量的声明语句(如:泛型声明)
ElementType.TYPE_USE:表示该注解能写在使用类型的任何语句中

@Repeatable(MyAnnotations.class)
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE_USE,ElementType.TYPE_PARAMETER})
public @interface MyAnnotation {
    String value();
}

class Geric<@MyAnnotation("hello") T>{//此处需要ElementType.TYPE_PARAMETER

    void show(){
        ArrayList<@MyAnnotation("String") String> list = new ArrayList();//此处需要ElementType.TYPE_USE
    }
}
使用反射获取元注解信息

前提:要求此注解的元注解Retention中声明的生命周期状态为RUNTIME
框架=注解+反射+设计模式

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容