[JavaSE] 第九章 枚举类与注解
9.1 枚举类的使用
-
枚举类的理解:类的对象只有有限个,确定的。我们称此类为枚举类
-
当需要定义一组常量时,强烈建议使用枚举类
-
如果枚举类中只有一个对象,则可以作为单例模式的实现方式
-
如何定义枚举类:
- 方式一:jdk5.0 之前,自定义枚举类
- 方式二:jdk5.0,可以使用 enum 关键字枚举类
-
使用 enum 关键字定义枚举类,定义的枚举类默认继承于 class java.lang.Enum 类
-
Enum 类中的常用方法:
- values() 方法:返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值
- valueOf(String str):可以把一个字符串转为对应的枚举类对象。要求字符串必须是枚举类对象的“名字”。如不是,会有运行时异常:
IllegalArgumentException
- toString():返回当前枚举类对象常量的名称
-
方式一
public class SeasonTest { public static void main(String[] args) { Season spring = Season.SPRING; System.out.println(spring); } } //自定义枚举类 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.其他诉求:获取枚举类对象的属性 public String getSeasonName() { return seasonName; } public String getSeasonDesc() { return seasonDesc; } //4.诉求:提供 toString() @Override public String toString() { return "Season{" + "seasonName='" + seasonName + '\'' + ", seasonDesc='" + seasonDesc + '\'' + '}'; } }
-
方式二
public class SeasonTest1 { public static void main(String[] args) { Season1 spring = Season1.SPRING; System.out.println(spring); System.out.println(Season1.AUTUMN); System.out.println(Season1.class.getSuperclass()); System.out.println(spring.toString()); Season1[] values = Season1.values(); for (Season1 value: values) { value.show(); System.out.println(value); } System.out.println("********************"); Thread.State[] values1 = Thread.State.values(); for (Thread.State value: values1) { System.out.println(value); } System.out.println("*****************"); //valueOf(String str):返回枚举类中对象名是 str 的对象 //如果没有 str 的枚举类对象,则抛出 IllegalArgumentException 异常 Season1 winter = Season1.valueOf("WINTER"); System.out.println(winter); spring.show(); winter.show(); } } interface Info { void show(); } //自定义枚举类 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.私有化类的构造器 Season1(String seasonName, String seasonDesc) { this.seasonName = seasonName; this.seasonDesc = seasonDesc; } //4.其他诉求:获取枚举类对象的属性 public String getSeasonName() { return seasonName; } public String getSeasonDesc() { return seasonDesc; } // @Override // public void show() { // System.out.println("这是一个季节"); // } //4.诉求:提供 toString() // @Override // public String toString() { // return "Season{" + // "seasonName='" + seasonName + '\'' + // ", seasonDesc='" + seasonDesc + '\'' + // '}'; // } }
9.2 注解的使用
- 注解 Annotation
- jdk5.0 新增
- Annotation 其实就是代码里的特殊标记, 这些标记可以在编译, 类加载, 运行时被读取,并执行相应的处理。通过使用 Annotation, 程序员可以在不改变原有逻辑的情况下, 在源文件中嵌入一些补充信息
- Annotation 可以像修饰符一样被使用, 可用于修饰包,类, 构造器, 方 法, 成员变量, 参数, 局部变量的声明,
这些信息被保存在 Annotation 的 “name=value” 对中 - 在 JavaSE 中,注解的使用目的比较简单,例如标记过时的功能,忽略警告等。在 JavaEE/Android 中注解占据了更重要的角色,例如用来配置应用程序的任何切面,代替 JavaEE 旧版中所遗留的繁冗代码和XML 配置等
- Annotation的使用示例:
- 示例一:生成文档相关的注解
- 示例二:在编译时进行格式检查(JDK 内置的三个基本注解)
@Override
: 限定重写父类方法, 该注解只能用于方法@Deprecated
: 用于表示所修饰的元素(类, 方法等)已过时。通常是因为所修饰的结构危险或存在更好的选择@SuppressWarnings
: 抑制编译器警告
- 示例三:跟踪代码依赖性,实现替代配置文件功能
- 如何址定义注解:参照
@SuppressWarnings
- 注解声明为:
@interface
- 内部定义成员,通常使用
value
表示 - 可以指定成员的默认值,使用
default
定义 - 如果自定义的注解没有成员,表明是一个标识作用
- 如果注解有成员,在使用注解时,需要指明成员的值,自定义注解必须配上注解的信息处理流程(使用反射)才有意义,自定义注解通常会指定
Retention
和Target
- 注解声明为:
- JDK 提供的 4 种元注解(元注解:对现有的注解进行解释说明的注解)
Retention
:指明所修饰的 Annotation 的生命周期:SOURCE\CLASS(默认行为)\RUNTIME 只有声明为 RUNTIME 生命周期的注解,才能通过反射获取Target
:用于指定被修饰Annotation
能用于修饰哪些程序元素Documented
:表示所修饰的注解在被javadoc
解析时,将被保留Inherited
:被它修饰的Annotation
将具有继承性
- JDK8 注解的新特性:可重复注解、类型注解
- 可重复注解:
- ① 在
MyAnnotation
上声明@Repeatable
,成员值为MyAnnotations.class
- ②
MyAnnotation
的Target
和Retention
等元注解 与MyAnnotations
的一致
- ① 在
- 类型注解
ElementType.TYPE_PARAMETER
表示该注解能写在类型变量的声明语句种(如:泛型声明)ElementType.TYPE_USE
表示该注解能写在适用类型的任何语句中
- 可重复注解:
public class AnnotationTest {
public static void main(String[] args) {
Student student = new Student();
student.show();
student.walk();
@SuppressWarnings("unused")
int i = 10;
}
@Test
public void getAnnotationTest() {
Class<Student> clazz = Student.class;
Annotation[] annotations = clazz.getAnnotations();
for (Annotation a: annotations) {
System.out.println(a);
}
}
}
//JDK8之前的写法
//@MyAnnotation(value = "Java")
//@MyAnnotations({@MyAnnotation(value = "Java"), @MyAnnotation(value = "SE")})
//JDK8
@MyAnnotation(value = "Java")
@MyAnnotation(value = "SE")
class Person {
private String name;
private int age;
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public void walk() {
System.out.println("人走路");
}
public void eat() {
System.out.println("人吃饭");
}
}
interface Info {
public abstract void show();
}
class Student extends Person implements Info {
@Override
public void walk() {
System.out.println("学生走路");
}
@Override
public void show() {
System.out.println("qwe");
}
}
class Generic<@MyAnnotation T> {
public void show() throws @MyAnnotation RuntimeException {
ArrayList<@MyAnnotation String> list = new ArrayList<>();
int num = (@MyAnnotation int)10L;
}
}
@Repeatable(MyAnnotations.class)
@Inherited
@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();
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· Ollama——大语言模型本地部署的极速利器
· 使用C#创建一个MCP客户端
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· ollama系列1:轻松3步本地部署deepseek,普通电脑可用
· 按钮权限的设计及实现