枚举
- 类的对象只有有限个,确定的。 使用场景:
- 星期: Monday(星期一)、 …、 Sunday(星期天)
- 性别: Man(男)、 Woman(女)
- 季节: Spring(春节)…Winter(冬天)
- 支付方式: Cash(现金)、 WeChatPay(微信)、 Alipay(支付宝)、 BankCard(银行卡)、 CreditCard(信用卡)
- 就职状态: Busy、 Free、 Vocation、 Dimission
- 订单状态: Nonpayment(未付款)、 Paid(已付款) 、 Delivered(已发货)、Return(退货)、 Checked(已确认) Fulfilled(已配货)、
- 线程状态:创建、就绪、运行、阻塞、死亡
- 当需要定义一组常量时,强烈建议使用枚举类 使用说明:
- 使用 enum 定义的枚举类默认继承了 java.lang.Enum类,因此不能再继承其他类
- 枚举类的构造器只能使用 private 权限修饰符
- 枚举类的所有实例必须在枚举类中显式列出(, 分隔 ; 结尾)。列出的实例系统会自动添加 public static final 修饰
- 必须在枚举类的第一行声明枚举类对象
- Enum类的主要方法:
- values()方法:返回枚举类型的对象数组。该方法可以很方便地遍历所有的枚举值。
- valueOf(String str):可以把一个字符串转为对应的枚举类对象。要求字符串必须是枚举类对象的“名字”。如不是,会有运行时异常:IllegalArgumentException。
- toString():返回当前枚举类对象常量的名称
使用常量表示线程状态
public class StateDem1 {
public static void main(String[] args) {
System.out.println(State.WAITING);
}
static class State {
public static final String NEW = "NEW";
public static final String RUNNABLE = "RUNNABLE";
public static final String BLOCKED = "BLOCKED";
public static final String WAITING = "WAITING";
public static final String TIMED_WAITING = "TIMED_WAITING";
public static final String TERMINATED = "TERMINATED";
// public static final String TERMINATED2 = "TERMINATED2";
}
}
使用枚举表示线程状态
public class StateDem2 {
int i;
static int i1;
public static void main(String[] args) {
//获取一个枚举对象
System.out.println(State.WAITING);
System.out.println(State.valueOf("BLOCKED"));
//循环枚举
State[] states = State.values();
for (int i = 0; i < states.length; i++) {
System.out.println(states[i]);
}
}
enum State {
//一个类State,中有6个这个类的静态对象
NEW,
RUNNABLE,
BLOCKED,
WAITING,
TIMED_WAITING,
TERMINATED;
}
}
订单状态的枚举
package a_enum;
/**
* 订单状态: Nonpayment(未付款)、 Paid(已付款) 、 Delivered(已发货)、Return(退货)、 Checked(已确认) Fulfilled(已配货)、
*/
public class StateDemo3 {
public static void main(String[] args) {
OrderState[] states = OrderState.values();
for (int i = 0; i < states.length; i++) {
System.out.println(states[i] + "-" + states[i].getValue() + "-" + states[i].getCode());
}
}
public enum OrderState {
//static OrderState NON_PAYMENT = new OrderState("未付款")
NON_PAYMENT(1, "未付款"),
PAID(2, "已付款"),
DELIVERED(3, "已发货"),
RETURN(4, "退货"),
CHECKED(5, "已确认"),
FULFILLED(6, "已配货");
private OrderState(int code, String value) {
this.code = code;
this.value = value;
}
private int code;
private String value;
public String getValue() {
return value;
}
public int getCode() {
return code;
}
}
}
注解
常见JDK注解
- @Override: 限定重写父类方法, 该注解只能用于方法
- @Deprecated: 用于表示所修饰的元素(类, 方法等)已过时。通常是因为所修饰的结构危险或存在更好的选择
- @SuppressWarnings: 抑制编译器警告
public class AnnotationTest{
public static void main(String[] args) {
@SuppressWarnings("unused")
int a = 10;
}
@Deprecated
public void print(){
System.out.println("过时的方法");
}
@Override
public String toString() {
return "重写的toString方法()";
}
}
JDK元注解
- JDK 的元 Annotation 用于修饰其他 Annotation 定义
- JDK5.0提供了4个标准的meta-annotation类型, 分别是:
- Retention
- Target
- Documented
- Inherited
@Retention
: 只能用于修饰一个 Annotation 定义, 用于指定该 Annotation 的生命周期, @Rentention 包含一个 RetentionPolicy 类型的成员变量, 使用@Rentention 时必须为该 value 成员变量指定值:RetentionPolicy.SOURCE
:在源文件中有效(即源文件保留) , 编译器直接丢弃这种策略的注释RetentionPolicy.CLASS
:在class文件中有效(即class保留) , 当运行 Java 程序时, JVM不会保留注解。 这是默认值RetentionPolicy.RUNTIME
:在运行时有效(即运行时保留) , 当运行 Java 程序时, JVM 会保留注释。程序可以通过反射获取该注释。
@Target
: 用于修饰 Annotation 定义, 用于指定被修饰的 Annotation 能用于修饰哪些程序元素。 @Target 也包含一个名为 value 的成员变量。- JDK1.8@Target新增类型如下:
ElementType.TYPE_PARAMETER
表示该注解能写在类型变量的声明语句中(如: 泛型声明) 。ElementType.TYPE_USE
表示该注解能写在使用类型的任何语句中。
- Java 8新增可重复注解:
@Documented
: 用于指定被该元 Annotation 修饰的 Annotation 类将被javadoc 工具提取成文档。 默认情况下, javadoc是不包括注解的。- 定义为Documented的注解必须设置Retention值为
RUNTIME
。
- 定义为Documented的注解必须设置Retention值为
@Inherited
: 被它修饰的 Annotation 将具有继承性。如果某个类使用了被@Inherited 修饰的 Annotation, 则其子类将自动具有该注解。- 比如:如果把标有@Inherited注解的自定义的注解标注在类级别上,子类则可以继承父类类级别的注解
得到方法的注解
要获取方法的注解,您可以使用 Java 的反射机制。下面是获取方法注解的一般步骤:
- .首先,通过反射获取相应的类对象或方法对象
- .使用
getAnnotations()
方法或getAnnotation(Class<? extends Annotation> annotationClass)
方法来获取该方法上的所有注解或指定注解的实例 - 处理获取到的注解对象,根据需要访问注解的属性或执行相应的操作。
以下是一个示例代码,演示如何获取方法的注解:
import java.lang.annotation.Annotation;
import java.lang.reflect.Method;
public class Main {
@MyAnnotation("Example")
public static void myMethod() {
// 方法体
}
public static void main(String[] args) throws NoSuchMethodException {
Method method = Main.class.getMethod("myMethod");
Annotation[] annotations = method.getAnnotations();
for (Annotation annotation : annotations) {
if (annotation instanceof MyAnnotation) {
MyAnnotation myAnnotation = (MyAnnotation) annotation;
String value = myAnnotation.value();
System.out.println("Method Annotation: " + value);
}
}
}
}
在上面的例子中,我们定义了一个自定义注解 MyAnnotation
,并将其应用于方法 myMethod()
。在 main()
方法中,我们首先使用 getMethod()
方法获取方法对象,然后使用 getAnnotations()
方法获取该方法上的所有注解。我们遍历这些注解,并筛选出我们感兴趣的 MyAnnotation
注解,然后访问注解的属性。
请注意,为了演示目的,我使用了一个自定义注解 MyAnnotation
。您可以根据您自己的需要修改和使用相应的注解。