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(); // 这是个属性
}