网站怎么做能赚钱吗,深圳市住建局造价站,网页制作教程电子书,重庆市工程建设信息网中项网1、背景 在项目中#xff0c;我们通常会具有同一特性的业务类定义一个顶层接口#xff0c;让业务类实现这个接口#xff0c;通过接口规范来管理这些类。我们将这些实现接口的业务类交托给Spring容器接口后#xff0c;有时候需要根据业务类型来选择动态选择对应的业务类阿里…1、背景 在项目中我们通常会具有同一特性的业务类定义一个顶层接口让业务类实现这个接口通过接口规范来管理这些类。我们将这些实现接口的业务类交托给Spring容器接口后有时候需要根据业务类型来选择动态选择对应的业务类阿里处理业务。这个时候就获取到这些业务类并进行管理在需要时取出对应的业务类处理业务。如何管理就是本期要介绍的内容。 以下是我定义的一个简单的顶层接口它有两个方法一个是提供类型的getType()方法一个是处理业务的hanlde()方法。我们的任务就是对其实现类进行管理当需要时可根据类型获取对应实现类。
public interface IBaseHandler {/*** 获取处理器类型* return 处理器类型*/int getType();/*** 处理业务* param t 业务数据* param T 业务数据类型*/T void handler(T t);
}
2、简单的管理方法 我们可以通过使用Autowired注解将所有实现了IBaseHandler接口的类注入到项目当中并在需要时遍历业务类对象获取对应的对象来处理业务。代码如下所示 Autowiredprivate ListIBaseHandler 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 MapInteger, IBaseHandler typeHandlerMap new HashMap();/*** 按照类型映射的处理器map*/private static MapClassIBaseHandler, IBaseHandler clazzHandlerMap new HashMap();/*** 初始化方法项目启动时调用该方法来初始化map* param applicationContext spring 上下文对象*/public static void init(ApplicationContext applicationContext) {ListIBaseHandler 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 - (ClassIBaseHandler) 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(ClassIBaseHandler 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 {Autowiredprivate ListIBaseHandler handlers;Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {// 调用处理器工具类初始化方法HandlerManager.init(applicationContext);}
}