Java 枚举对象获取Class对象

本文探讨了在Java中使用枚举类型时遇到的问题,特别是当枚举项重写方法时,如何正确地通过注解获取枚举描述字段。文章详细分析了问题原因,并给出了使用getDeclaringClass()方法来解决这一问题的有效方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

背景

       在写状态机框架的时候,在状态枚举类型上面加了注解,当通过 getClass() 得到 Class 对象得到注解时,有些情况得不到注解。发现是自己对 Java 枚举了解不够导致,特记录一下。

定义一个方法 getStateDescField(S s) :从注解@StateConfig指定的字段名称,获取传入 s 对应的字段值。

public static <S> Object getStateDescField(S s) {
        Class clazz = s.getClass();
        if (!clazz.isAnnotationPresent(StateConfig.class)) {
            return null;
        }
        StateConfig stateConfig = (StateConfig) clazz.getAnnotation(StateConfig.class);
        String descField = stateConfig.descField();
        try {
            Field field = clazz.getDeclaredField(descField);
            field.setAccessible(true);
            return field.get(s);
        } catch (Exception e) {
            e.printStackTrace();
            return null;
        }
    }

  入参 S 传入枚举对象,看一下我们使用的一个状态枚举:

@StateConfig(descField = "desc")
public enum OrderStatusEnum {
    CLOSED(-10, "订单关闭"),
    INIT(10, "订单生成"){
        @Override
        public boolean cancelable(Integer openTicketNode) {
            return true;
        }
    },
    PRE_CHECK(15,"待平台预审"){
        @Override
        public boolean cancelable(Integer openTicketNode) {
            return true;
        }
    },
    WAITTING_TO_CHECK(20, "待机构审批"){
        @Override
        public boolean cancelable(Integer openTicketNode) {
            return true;
        }
    },

    WAITTING_TO_LOAN(70, "待机构放款"){
        @Override
        public boolean cancelable(Integer openTicketNode) {
            return true;
        }
    },

    LOANED(80, "已放款");

    private Integer code;
    private String desc;

    OrderStatusEnum(Integer code, String desc) {
        this.code = code;
        this.desc = desc;
    }

    public boolean cancelable(Integer openTicketNode){
        return false;
    }

}

       最后发现,只有 CLOSED 和 LOANED 两个枚举值成功读出了 desc 描述字段值,其余状态枚举读出的都是 null 。很明显,读不出 desc 字段的枚举值INIT、PRE_CHECK 等都重写了方法 cancelable (Integer openTicketNode),应该就是这里导致出现的问题。

Debug

debug 代码发现,CLOSED 枚举进入 getStateDescField(S s) 方法,得到的 Class 对象是OrderStatusEnum:

但是 INIT 枚举值执行 getStateDescField(S s) 方法,得到的 Class 对象是OrderStatusEnum中的内部类 OrderStatusEnum$1:

而 OrderStatusEnum$1 上是没有 @StateConfig 的,所以返回了 null。

结论:

       这就是为什么推荐 枚举类型获取Class对象时,推荐使用 getDeclaringClass() 方法【java.lang.Enum】的原因了:

    /**
     * Returns the Class object corresponding to this enum constant's
     * enum type.  Two enum constants e1 and  e2 are of the
     * same enum type if and only if
     *   e1.getDeclaringClass() == e2.getDeclaringClass().
     * (The value returned by this method may differ from the one returned
     * by the {@link Object#getClass} method for enum constants with
     * constant-specific class bodies.)
     *
     * @return the Class object corresponding to this enum constant's
     *     enum type
     */
    @SuppressWarnings("unchecked")
    public final Class<E> getDeclaringClass() {
        Class<?> clazz = getClass();
        Class<?> zuper = clazz.getSuperclass();
        return (zuper == Enum.class) ? (Class<E>)clazz : (Class<E>)zuper;
    }

       getDeclaringClass() 会先判断上一级 zuper 是不是 Enum.class。我们通过代码测试,可以知道 CLOSED 和 INIT 对应 getClass()和getSuperclass()方法的结果:

    public static void main(String[] args) {
        System.out.println(OrderStatusEnum.CLOSED.getClass());
        System.out.println(OrderStatusEnum.CLOSED.getClass().getSuperclass());
        System.out.println(OrderStatusEnum.INIT.getClass());
        System.out.println(OrderStatusEnum.INIT.getClass().getSuperclass());
        System.out.println(OrderStatusEnum.INIT.getClass().getSuperclass().getSuperclass());
    }

输出结果:

class OrderStatusEnum
class java.lang.Enum
class OrderStatusEnum$1
class OrderStatusEnum
class java.lang.Enum

      可以看到,实现了方法的枚举值INIT需要调用两次 getSuperclass() 方法才能得到 java.lang.Enum 的Class对象,调用一次 getSuperclass() 方法刚好可以得到OrderStatusEnum的Class 对象。而 CLOSED 枚举值直接getClass()就得到OrderStatusEnum的Class 对象。所以 getDeclaringClass() 方法能确保 INIT 和CLOSED 获取到的都是 OrderStatusEnum的Class 对象。

       另外我们通过命令编译 javac OrderStatusEnum.java,确实可以看到 INIT、PRE_CHECK、WAITTING_TO_CHECK、WAITTING_TO_LOAN 被编译成了四个内部类:

使用 javap -c 命令可以看到 OrderStatusEnum 继承了 Enum 类:

       INIT 等继承覆盖了 OrderStatusEnum 中的 cancelable 方法,而且上面 INIT.getClass().getSuperClass 为 OrderStatusEnum ,所以 INIT 对应的 OrderStatusEnum$1 应该继承了 OrderStatusEnum,但是一时不知道怎么反编译 OrderStatusEnum$1查看验证,麻烦有知道的告诉一下哈~

### C语言结构体初始化方法 在C语言中,结构体是一种复合数据类型,可以用来存储不同类型的数据。为了正确使用结构体,必须对其进行初始化。以下是常见的几种初始化方法及其示例。 #### 1. **顺序初始化** 按照成员定义的顺序依次赋初值给各个成员。如果只需要初始化部分成员,则未初始化的部分会保持默认值(通常是0或空字符)。需要注意的是,一旦某个成员被跳过,后续成员也不能再初始化[^3]。 ```c struct Test { int a; int b; char c; double d; }; // 定义并初始化结构体变量test1 struct Test test1 = {1, 2}; // 此处a=1,b=2,c和d取默认值(通常为0) ``` #### 2. **指定成员初始化** 从C99标准起支持通过指定成员名的方式进行初始化。这种方式更加灵活,允许任意次序初始化,并且可以选择性忽略某些成员[^4]。 ```c struct Test { int a; int b; char c; double d; }; // 使用指定成员方式初始化结构体变量test2 struct Test test2 = {.b = 3, .c = 'X', .d = 78.5}; // 这里a未显式设置,默认为0 ``` #### 3. **动态内存分配后的初始化** 当利用`malloc()`或其他动态分配函数创建结构体实例时,需手动完成其内部各字段的设定工作。此时无法直接采用大括号列表形式来实现整体初始化,而应逐一访问每个域来进行具体数值赋予操作。 ```c #include <stdlib.h> struct Student { char c; int score; char* name; }; void initializeStudent(struct Student *stu){ stu->c = 'A'; stu->score = 95; stu->name = "Tom"; } int main(){ struct Student *pStu = (struct Student *) malloc(sizeof(struct Student)); if(pStu != NULL){ initializeStudent(pStu); printf("c = %c, score = %d, name = %s\n", pStu->c, pStu->score, pStu->name); free(pStu); } } ``` 上述代码展示了如何先申请一块堆空间用于存放一个新的学生记录对象,之后调用辅助函数对其属性加以配置最后打印出来验证效果同时记得释放资源防止泄露问题发生。 ### 总结 综上所述,在实际开发过程中可以根据需求选用不同的初始化策略以满足特定场景下的应用要求。无论是静态声明还是运行期间构建出来的实体都可以找到合适的办法去处理它们各自的初始状态调整事宜[^1][^2].
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值