SpringBoot顶层接口实现类注入项目的方法示例

 更新时间:2024年06月17日 08:55:11   作者:木星mx  
本文主要介绍了SpringBoot顶层接口实现类注入项目的方法示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1、背景

在项目中,我们通常会具有同一特性的业务类定义一个顶层接口,让业务类实现这个接口,通过接口规范来管理这些类。我们将这些实现接口的业务类交托给Spring容器接口后,有时候需要根据业务类型来选择动态选择对应的业务类阿里处理业务。这个时候就获取到这些业务类并进行管理,在需要时取出对应的业务类处理业务。如何管理就是本期要介绍的内容。

以下是我定义的一个简单的顶层接口,它有两个方法,一个是提供类型的getType()方法,一个是处理业务的hanlde()方法。我们的任务就是对其实现类进行管理,当需要时可根据类型获取对应实现类。

public interface IBaseHandler {
    /**
     * 获取处理器类型
     * @return 处理器类型
     */
    int getType();

    /**
     * 处理业务
     * @param t 业务数据
     * @param <T> 业务数据类型
     */
    <T> void handler(T t);
}

2、简单的管理方法

我们可以通过使用@Autowired注解将所有实现了IBaseHandler接口的类注入到项目当中,并在需要时遍历业务类对象,获取对应的对象来处理业务。代码如下所示:

    @Autowired
    private List<IBaseHandler> handlers;
     /**
     * 处理业务
     * @param type 业务类型
     * @param data 业务数据
     * @param <T> 业务数据类型
     */
    public <T> void handle(int type, T data) {
        handlers.stream().filter(handler -> handler.getType() == type).findAny().orElseThrow(() -> {
            // 获取不到业务类对象时打印日志并抛出异常
            log.error("Failed to get handler, type:{}", type);
            throw new NoSuchElementException("No such handler");
        }).handler(data);
    }

3、更好的管理方法

1、简单管理方法的弊端

上面的简单管理方法用起来方便,但是有两个弊端:
(1)耦合度高,所有需要使用该接口的地方都需要进行注入再遍历的过程。
(2)性能较差,每次执行业务之前都需要遍历一次列表。

2、解决

我们可以使用一个工具类,提供静态方法来获取业务类。这样所有需要获取业务类的地方,就都可以通过该工具类一步获取到所需的业务类。代码如下:

1、工具类代码

@Slf4j
public class HandlerManager {
    /**
     * 按照type映射的处理器map
     */
    private static Map<Integer, IBaseHandler> typeHandlerMap = new HashMap<>();
    /**
     * 按照类型映射的处理器map
     */
    private static Map<Class<IBaseHandler>, IBaseHandler> clazzHandlerMap = new HashMap<>();

    /**
     * 初始化方法,项目启动时调用该方法来初始化map
     * @param applicationContext spring 上下文对象
     */
    public static void init(ApplicationContext applicationContext) {
        List<IBaseHandler> handlers = new ArrayList<>();
        applicationContext.getBeansOfType(IBaseHandler.class).forEach((name, obj) -> handlers.add(obj));
        // 为了方便,两次循环构建map,一次循环也可以解决,不过人为定义的handler数量不多,一次循环性能提升不大
        typeHandlerMap = handlers.stream().collect(Collectors.toMap(IBaseHandler::getType, obj -> obj));
        clazzHandlerMap = handlers.stream().collect(Collectors.toMap(obj -> (Class<IBaseHandler>) obj.getClass(), obj -> obj));
    }

    public IBaseHandler getHandlerByType(int type) {
        return Optional.ofNullable(typeHandlerMap.get(type)).orElseThrow(() -> {
            // 获取不到处理器打印日志并抛出异常
            log.info("Failed to get handler, type:{}", type);
            throw new NoSuchElementException("No such  handler error");
        });
    }

    public IBaseHandler getHandlerByClass(Class<IBaseHandler> clazz) {
        return Optional.ofNullable(clazzHandlerMap.get(clazz)).orElseThrow(() -> {
            // 获取不到处理器打印日志并抛出异常
            log.info("Failed to get handler, clazz:{}", clazz);
            throw new NoSuchElementException("No such  handler error");
        });
    }


}

2、调用工具类的init方法

由于工具类提供的是静态方法,因此无法通过注入的方式来获取到所有的IBaseHanlder,所以需要借助其他可注入IBaseHanlder的类来初始化管理对象,以下是方法:

/**
 * 项目初始化类
 * 继承ApplicationContextAware,实现setApplicationContext,可获取ApplicationContext来获取上下文
 */
@Component
public class ApplicationInit implements ApplicationContextAware {

    @Autowired
    private List<IBaseHandler> handlers;

    @Override
    public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
        // 调用处理器工具类初始化方法
        HandlerManager.init(applicationContext);
    }
}

到此这篇关于SpringBoot顶层接口实现类注入项目的方法示例的文章就介绍到这了,更多相关SpringBoot类注入内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家! 

相关文章

  • java基础--自己动手实现一个LRU

    java基础--自己动手实现一个LRU

    这篇文章主要介绍了运用方案如何实现LUR,文章中通过代码讲解的非常详细,对大家的工作或学习有一定的参考价值,感兴趣的朋友可以参考一下
    2021-08-08
  • Java中的Caffeine加载与驱逐策略详解

    Java中的Caffeine加载与驱逐策略详解

    这篇文章主要介绍了Java中的Caffeine加载与驱逐策略详解,Caffeine是基于Java 8的高性能缓存库,可提供接近最佳的命中率,Caffeine与ConcurrentMap相应,但是不完全相同,本文主要介绍Caffeine,需要的朋友可以参考下
    2023-10-10
  • Mybatis Interceptor 拦截器的实现

    Mybatis Interceptor 拦截器的实现

    这篇文章主要介绍了Mybatis Interceptor 拦截器的实现,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • SpringBoot整合Web之AOP配置详解

    SpringBoot整合Web之AOP配置详解

    面向切面编程(aspect-oriented programming,AOP)主要实现的目的是针对业务处理过程中的切面进行提取,诸如日志、事务管理和安全这样的系统服务,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率
    2022-08-08
  • java实现随机数生成器

    java实现随机数生成器

    这篇文章主要为大家详细介绍了java实现随机数生成器,随机数生成小程序,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • 解析Java的Hibernate框架中的持久化类和映射文件

    解析Java的Hibernate框架中的持久化类和映射文件

    这篇文章主要介绍了Java的Hibernate框架中的持久化类和映射文件,Hibernate是Java的SSH三大web开发框架之一,需要的朋友可以参考下
    2015-12-12
  • springboot中如何使用@Async方法

    springboot中如何使用@Async方法

    这篇文章主要介绍了springboot中如何使用@Async方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-08-08
  • 使用maven打包生成doc文档和打包源码

    使用maven打包生成doc文档和打包源码

    这篇文章主要介绍了使用maven打包生成doc文档和打包源码的实现,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-07-07
  • Java设计模式七大原则之里氏替换原则详解

    Java设计模式七大原则之里氏替换原则详解

    在面向对象的程序设计中,里氏替换原则(Liskov Substitution principle)是对子类型的特别定义。本文将为大家详细介绍Java设计模式七大原则之一的里氏替换原则,需要的可以参考一下
    2022-02-02
  • Spring事务失效的9大场景与解决方法

    Spring事务失效的9大场景与解决方法

    在日常开发中,我们经常使用Spring事务,这篇文章主要来和大家聊聊Spring事务失效的 9 种场景,并提供了相应的解决方法,有需要的小伙伴可以参考一下
    2025-04-04

最新评论