java通过反射获取加了某个注解的所有的类

简介: java通过反射获取加了某个注解的所有的类

一、前言


有时候我们会碰到这样的情况:


有n个场景,每个场景都有自己的逻辑,即n个处理逻辑,


这时候我们就需要通过某个参数的值代表这n个场景,然后去加载每个场景不同的bean对象,即不同的类,这些类中都有一个同名的方法,但是里面的逻辑不同,类似策略模式、工厂模式等


假设这样的场景,银行卡分几种类型,比如普通会员,黄金会员,黑卡会员......


普通会员、黄金会员和黑卡的权限不一样,消费能力不一样等等内容,这里我们就以消费的能力举例,普通会员只能转账5万元,黄金只能转账20万元,黑卡100万元,


我们有3各类,分别是Putong.class,Huangjin.class,Heika.class,这三个类里面有一个同样的方法,名字叫:checkPermissions()方法,三个类上都加了一个自定义注解@MemberTypeDefinition


二、代码demo参考

1、自定义注解

package com.zygxsq.test;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
/**
 * 自定义注解
 *
 **/
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface MemberTypeDefinition {
    /**
     * 数据源默认执行方法
     */
    public static final String DEFAULT_DAO_METHOD_NAME = "checkPermissions";
    /**
     * 数据源执行的方法名称, 默认为getData, 该方法必须只有一个参数{@link BaseDataSpec}, 返回结果必须是{@link BaseResult}, 可见性必须是public
     *
     * @return 方法名称
     */
    String value() default DEFAULT_DAO_METHOD_NAME;
}


2、基础model类

Putong.class

package com.zygxsq.test;
import org.springframework.stereotype.Component;
/**
 * 普通会员处理逻辑
 *
 **/
@MemberTypeDefinition
@Slf4j
@Component
public class Putong {
    public BaseResult checkPermissions(BaseDataSpec baseDataSpec) {
        // 省略中间的一些处理逻辑
        // ......
        // 省略中间的一些处理逻辑
        // 最后封装返回结果
        BaseResult baseResult = new BaseResult();
        baseResult.setResponse("我是普通会员");
        return baseResult;
    }
}

Huangjin.class

package com.zygxsq.test;
import org.springframework.stereotype.Component;
/**
 * 黄金会员处理逻辑
 *
 **/
@MemberTypeDefinition
@Slf4j
@Component
public class Huangjin {
    public BaseResult checkPermissions(BaseDataSpec baseDataSpec) {
        // 省略中间的一些处理逻辑
        // ......
        // 省略中间的一些处理逻辑
        // 最后封装返回结果
        BaseResult baseResult = new BaseResult();
        baseResult.setResponse("我是黄金会员");
        return baseResult;
    }
}

Heika.class

package com.zygxsq.test;
import org.springframework.stereotype.Component;
/**
 * 黑卡会员处理逻辑
 *
 **/
@MemberTypeDefinition
@Slf4j
@Component
public class Heika {
    public BaseResult checkPermissions(BaseDataSpec baseDataSpec) {
        // 省略中间的一些处理逻辑
        // ......
        // 省略中间的一些处理逻辑
        // 最后封装返回结果
        BaseResult baseResult = new BaseResult();
        baseResult.setResponse("我是黑卡会员");
        return baseResult;
    }
}

3、入参和返回参数对象

入参对象

package com.zygxsq.test;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.experimental.SuperBuilder;
import java.io.Serializable;
import java.util.List;
/**
 * 入参数据对象
 *
 **/
@Data
@SuperBuilder
@NoArgsConstructor
@AllArgsConstructor
public class BaseDataSpec implements Serializable {
    /**
     * serialVersionUID
     */
    private static final long serialVersionUID = -5669150877852962345L;
    /**
     * id
     */
    private String id;
    /**
     * 名字
     */
    private String name;
}

返回结果对象

package com.zygxsq.test;
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;
import java.io.Serializable;
/**
 * 返回结果对象
 *
 **/
@Data
@Builder
@NoArgsConstructor
@AllArgsConstructor
public class BaseResult implements Serializable {
    /**
     * serialVersionUID
     */
    private static final long serialVersionUID = 7510262928468530569L;
    private String response;
}

4、反射核心代码

那如何通过反射进行加载呢,BaseDataAnnotationApplication.class

package com.zygxsq.test;
import com.google.common.collect.Maps;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Map;
/**
 * 启动加载
 *
 **/
@Slf4j
@Component
public class BaseDataAnnotationApplication implements ApplicationContextAware, InitializingBean {
    private ApplicationContext applicationContext;
    /**
     * <className,V>
     */
    private final Map<String, Object> loadMap = Maps.newConcurrentMap();
    /**
     * 特殊人群表缓存 <id,1> ,获取 SpecialPeople.class里面所有的表名
     */
    public static Map<String,Object> dmTablesMap = Maps.newConcurrentMap();
    @Override
    public void afterPropertiesSet() throws Exception {
        // 获取加了 MemberTypeDefinition 注解的源表bean
        loadSourceDefinition();
        // 获取SpecialPeople.class里面所有的表名
        loadSpecialMap();
    }
    private void loadSourceDefinition() throws Exception{
        Map<String, Object> beans = applicationContext.getBeansWithAnnotation(MemberTypeDefinition.class);
        for (final Object serviceObject : beans.values()) {
            final Class<? extends Object> calcuteClass = serviceObject.getClass();
            MemberTypeDefinition annotation = calcuteClass.getAnnotation(MemberTypeDefinition.class);
            if (null == annotation) {
                log.error("类: {} 注解缺失", calcuteClass);
                continue;
            }
            loadMap.put(calcuteClass.getName(), serviceObject);
        }
    }
    /**
     * 获取SpecialPeople.class里面所有的表名
     * @throws Exception
     */
    private void loadSpecialMap() throws Exception {
        dmTablesMap = Maps.newConcurrentMap();
        Field[] declaredFields = SpecialPeople.class.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            Class<?> type = declaredField.getType();
            String typeName = declaredField.getGenericType().getTypeName();
            Class<?> aClass = Class.forName(typeName);
            if (type == People.class) {
                People people = (People) declaredField.get(aClass);
                String id = people.getId().trim();
                dmTablesMap.put(id, "1");
            }
        }
    }
    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        this.applicationContext = applicationContext;
    }
    public <T> BaseResult buildData(Class<T> clazz, BaseDataSpec baseDataSpec) throws Exception{
        // 获取执行方法名, 默认为getData
        String methodName;
        MemberTypeDefinition annotation = clazz.getAnnotation(MemberTypeDefinition.class);
        if (annotation == null || StringUtils.isBlank(annotation.value())) {
            methodName = MemberTypeDefinition.DEFAULT_DAO_METHOD_NAME;
        } else {
            methodName = annotation.value();
        }
        Method method;
        Object bean = loadMap.get(clazz.getName());
        BaseResult result = null;
        try {
            method = bean.getClass().getMethod(methodName, BaseDataSpec.class);
            result = (BaseResult) method.invoke(bean, baseDataSpec);
        } catch (NoSuchMethodException e) {
            throw new Exception(clazz.getName()+"未找到执行方法:"+methodName);
        } catch (Exception e2) {
            throw new Exception(clazz.getName()+"未找到执行方法:"+methodName);
        }
        return result;
    }
    public <T> BaseResult buildData(Class<?> sourceClass) {
        return null;
    }
}

通过上面的application,就可以加对象加载到缓存里了,然后我们直接调用即可

5、测试接口

package com.zygxsq.test;
import com.zygxsq.test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/users")
public class TestController {
    @Autowired
    private BaseDataAnnotationApplication baseDataAnnotationApplication;
    @RequestMapping("/getUserType")
    public String getUserType() {
        // ...
        // 省略查询表等逻辑
        // ...
        Class<test.Putong> putongClass = Putong.class;
        BaseDataSpec baseDataSpec = BaseDataSpec.builder()
                .id("888888")
                .build();
        BaseResult baseResult = baseDataAnnotationApplication.buildData(putongClass,baseDataSpec);
        String response = baseResult.getResponse();
        System.out.println(response);
        return response;
    }
}

6、代码结构

所有代码结构,大致是这样,具体的一些代码可以放在不同的package下面,我这里仅仅是让大家可以直观的看到有这些类

相关文章
|
14天前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
50 7
|
19天前
|
Java 编译器 数据库
Java 中的注解(Annotations):代码中的 “元数据” 魔法
Java注解是代码中的“元数据”标签,不直接参与业务逻辑,但在编译或运行时提供重要信息。本文介绍了注解的基础语法、内置注解的应用场景,以及如何自定义注解和结合AOP技术实现方法执行日志记录,展示了注解在提升代码质量、简化开发流程和增强程序功能方面的强大作用。
55 5
|
26天前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
44 8
|
25天前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
26天前
|
Java 开发者
在 Java 中,一个类可以实现多个接口吗?
这是 Java 面向对象编程的一个重要特性,它提供了极大的灵活性和扩展性。
59 1
|
1月前
|
安全 Java
Java多线程集合类
本文介绍了Java中线程安全的问题及解决方案。通过示例代码展示了使用`CopyOnWriteArrayList`、`CopyOnWriteArraySet`和`ConcurrentHashMap`来解决多线程环境下集合操作的线程安全问题。这些类通过不同的机制确保了线程安全,提高了并发性能。
|
1月前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
114 4
|
1月前
|
Java 编译器 开发者
Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面
本文探讨了Java异常处理的最佳实践,涵盖理解异常类体系、选择合适的异常类型、提供详细异常信息、合理使用try-catch和finally语句、使用try-with-resources、记录异常信息等方面,帮助开发者提高代码质量和程序的健壮性。
58 2
|
1月前
|
Java Android开发
Eclipse 创建 Java 类
Eclipse 创建 Java 类
26 0
|
2月前
|
IDE Java 编译器
java的反射与注解
java的反射与注解
17 0