一、枚举类
JDK1.5之前需要自定义枚举类
枚举类对象的属性不应允许被改动, 所以应该使用 private final 修饰
若枚举只有一个成员, 则可以作为一种单例模式的实现方式
public class TestSeason {
public static void main(String args[]){
Season spring = Season.SPRING;
System.out.println(spring);
spring.show();
System.out.println(spring.getSeasonName());
}
}
//枚举类
class Season {
// 1.提供类的属性,声明为private final
private final String seasonName;
private final String seasonDesc;
// 2.声明为final的属性,在构造器中初始化
private Season(String seasonName,String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
// 3.通过公共的方法来调用属性
public String getSeasonName(){
return seasonName;
}
public String getSeasonDesc(){
return seasonDesc;
}
// 4.创建为类的对象:将类的对象声明为 public static final
public static final Season SPRING = new Season("spring","春暖花开");
public static final Season SUMMER = new Season("summer","夏日炎炎");
public static final Season AUTUMN = new Season("autumn","秋高气爽");
public static final Season WINTER = new Season("winter","白雪皑皑");
@Override
public String toString() {
return "Season [seasonName=" + seasonName + ", seasonDesc=" + seasonDesc + "]";
}
public void show(){
System.out.println("这是一个季节");
}
}JDK 1.5 新增的 enum 关键字用于定义枚举类
必须在枚举类的第一行声明枚举类对象。
使用 enum 定义的枚举类默认继承了 java.lang.Enum 类。
列出的实例系统会自动添加 public static final 修饰。
枚举类的主要方法:
values()方法:
- 返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值。
valueOf(String str):
- 可以把一个字符串转为对应的枚举类对象。要求字符串必须是枚举类对象的“名字”。如不是,会有运行时异常。
若需要每个枚举值在调用实现的接口方法呈现出不同的行为方式, 则可以让每个枚举值分别来实现该方法。
public class TestSeason1 {
public static void main(String args[]){
Season1 spring = Season1.SPRING;
System.out.println(spring);
spring.show();
System.out.println(spring.getSeasonName());
//1.values()
Season1 seasons[] = Season1.values();
for(int i=0; i < seasons.length; i++){
System.out.println(seasons[i]);
}
//2.valueOf(String name) 要求传入的形参name是枚举类对象的名字。否则,报java.lang.IllegalArgumentException异常
String str = "SPRING";
Season1 sea = Season1.valueOf(str);
System.out.println(sea);
Thread.State states[] = Thread.State.values();
for(int i=0; i<states.length; i++){
System.out.println(states[i]);
}
}
}
interface Info{
void show();
}
//枚举类
enum Season1 implements Info{
SPRING("spring","春暖花开"){
public void show(){
System.out.println("春天在哪里?");
}
},
SUMMER("summer","夏日炎炎"){
public void show(){
System.out.println("生如夏花");
}
},
AUTUMN("autumn","秋高气爽"){
public void show(){
System.out.println("秋天是分手的季节");
}
},
WINTER("winter","白雪皑皑"){
public void show(){
System.out.println("冬天里的一把火");
}
};
private final String seasonName;
private final String seasonDesc;
private Season1(String seasonName,String seasonDesc){
this.seasonName = seasonName;
this.seasonDesc = seasonDesc;
}
public String getSeasonName(){
return seasonName;
}
public String getSeasonDesc(){
return seasonDesc;
}
@Override
public String toString() {
return "Season [seasonName=" + seasonName + ", seasonDesc=" + seasonDesc + "]";
}
// public void show(){
// System.out.println("这是一个季节");
// }
}二、注解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 value() default "hello";
}- 没有成员定义的 Annotation 称为标记; 包含成员变量的 Annotation 称为元数据 Annotation
JDK 的元 Annotation
- JDK 中的元Annotation用于修饰其他Annotation定义
- JDK提供了专门在注解上的注解类型,分别是:
- @Retention: 只能用于修饰一个 Annotation 定义, 用于指定该 Annotation 可以保留多长时间(注解的生命周期), @Rentention 包含一个 RetentionPolicy 类型的成员变量, 使用 @Rentention 时必须为该 value 成员变量指定值:
- RetentionPolicy.SOURCE:在源文件中有效,被编译器丢弃。
- RetentionPolicy.CLASS: 在编译器生成的字节码文件中有效,但在运行时会被处理类文件的 JVM 丢弃。 这是默认值
- RetentionPolicy.RUNTIME:到运行时都有效。这也是注解生命周期中最常用的一种策略,它允许程序通过反射的方式访问注解,并根据注解的定义执行相应的代码。
- @Target: 用于修饰 Annotation 定义, 用于指定被修饰的 Annotation 能用于修饰哪些程序元素. @Target 也包含一个名为 value 的成员变量.
- 截止到 Java 9,注解的类型一共有 11 种,定义在 ElementType 枚举中:
- 1)TYPE:用于类、接口、注解、枚举
- 2)FIELD:用于字段(类的成员变量),或者枚举常量
- 3)METHOD:用于方法
- 4)PARAMETER:用于普通方法或者构造方法的参数
- 5)CONSTRUCTOR:用于构造方法
- 6)LOCAL_VARIABLE:用于变量
- 7)ANNOTATION_TYPE:用于注解
- 8)PACKAGE:用于包
- 9)TYPE_PARAMETER:用于泛型参数
- 10 TYPE_USE:用于声明语句、泛型或者强制转换语句中的类型
- 11)MODULE:用于模块
- @Documented: 用于指定被该元 Annotation 修饰的 Annotation 类将被 javadoc 工具提取成文档。定义为Documented的注解必须设置Retention值为RUNTIME。
- @Inherited: 被它修饰的 Annotation 将具有继承性.如果某个类使用了被 @Inherited 修饰的 Annotation, 则其子类将自动具有该注解。实际应用中,使用较少
Annotation应用Demo
我们要利用注解 实现一个自定义序列化工具类,有注解的对象的属性才被序列化进去。
自定义注解类:
/**
* 作用:用来标记对象在序列化成 JSON 的时候要不要包含这个字。
* JsonField 注解的生命周期是 RUNTIME,也就是运行时有效。
* JsonField 注解装饰的目标是 FIELD,也就是针对字段的(类的成员变量)。
* 创建注解需要用到 @interface 关键字。
* JsonField 注解有一个参数,名字为 value,类型为 String,默认值为一个空字符串(它允许我们可以直接使用 @JsonField,而无需指定参数的名和值)。
*/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface JsonField {
public String value() default "";
}实体类:
@Data
@AllArgsConstructor
public class Writer {
private int age;
@JsonField("writerName")
private String name;
@JsonField
private String bookName;
}序列化工具类:
public class JsonSerializer {
public static String serialize(Object object) throws IllegalAccessException {
Class<?> objectClass = object.getClass();
Map<String, String> jsonElements = new HashMap<>();
// 获取对象声明的所有字段
for (Field field : objectClass.getDeclaredFields()) {
// 将反射对象的可访问性设置为 true,防止private 字段无法获取
field.setAccessible(true);
if (field.isAnnotationPresent(JsonField.class)) {
jsonElements.put(getSerializedKey(field), (String) field.get(object));
}
}
return toJsonString(jsonElements);
}
/**
* 获取字段上注解的值,如果注解的值是空的,则返回字段名
* @param field
* @return
*/
private static String getSerializedKey(Field field) {
String annotationValue = field.getAnnotation(JsonField.class).value();
if (annotationValue.isEmpty()) {
return field.getName();
} else {
return annotationValue;
}
}
private static String toJsonString(Map<String, String> jsonMap) {
String elementsString = jsonMap.entrySet()
.stream()
.map(entry -> "\"" + entry.getKey() + "\":\"" + entry.getValue() + "\"")
.collect(Collectors.joining(","));
return "{" + elementsString + "}";
}
}测试入口:
public class JsonFieldTest {
public static void main(String[] args) throws IllegalAccessException {
Writer cmower = new Writer(16,"无关风月","Java从入门到放弃");
System.out.println(JsonSerializer.serialize(cmower));
}
}运行结果:
{"bookName":"Java从入门到放弃","writerName":"无关风月"}从结果上来看:
1)Writer 类的 age 字段没有装饰 @JsonField 注解,所以没有序列化。
2)Writer 类的 name 字段装饰了 @JsonField 注解,并且显示指定了字符串“writerName”,所以序列化后变成了 writerName。
3)Writer 类的 bookName 字段装饰了 @JsonField 注解,但没有显式指定值,所以序列化后仍然是 bookName。
参考:
















