摘要:動態代理反射注解優化代碼一動態代理提供接口默認實現我們拋出問題,并且提出解決問題的第一步的方法。重寫動態代理類,實現通過的查找出傳入的所有泛型的實現下一篇動態代理反射注解優化代碼三注解
SpringBoot 動態代理|反射|注解|AOP 優化代碼(一)-動態代理提供接口默認實現 我們拋出問題,并且提出解決問題的第一步的方法。下面我們繼續深入,動態代理和反射繼續解決我們的問題。
改動代碼結構新增一個HandlerRougter接口,其目的就是替代上一篇的DeviceHandlerRouter
public interface HandlerRouter{ T getHandler(Integer env,Object... args); }
其中T是具體的業務接口。下面實現DeviceHandler的HandlerRouter:
public interface DeviceHandlerRouter extends HandlerRouter{ }
那么上層代碼的調用方式將會類似下面的代碼:
DeviceHandlerRouter deviceHandlerRouter = ... deviceHandlerRouter.getHandler(...). remoteAddBatch(...)反射+動態代理
前面說過,每增加一種接口調用,就需要重新實現xxxHandlerRouter,那么下面我們通過動態代理和反射提供DeviceHandler的默認實現。
1.通過反射獲取HandlerRouter
首先加入下面的依賴
org.reflections reflections 0.9.10
/** * 通過反射掃描出所有HandlerRouter的子類 * @return */ private Set> getHandlerRouterClasses() { Reflections reflections = new Reflections( "package.name.*", new TypeAnnotationsScanner(),//注解掃描,本節用不到 new SubTypesScanner() ); return reflections.getSubTypesOf(HandlerRouter.class); }
Set> classes = getHandlerRouterClasses(); //獲取HandlerRouter的子接口的泛型Class 例如:DeviceHandlerRouter接口的DeviceHandler for (Class> clazz : classes) { //clazz 對應DeviceHandlerRouter.class Type[] types = clazz.getGenericInterfaces(); ParameterizedType type = (ParameterizedType) types[0]; //typeName對應DeviceHandlerRouter extends HandlerRouter 中的DeviceHandler.class String typeName = type.getActualTypeArguments()[0].getTypeName(); }
2.SpringBoot ApplicationContext 獲取注入的bean
修改上一篇中實現的動態代理類,我們想要實現:“能夠根據傳入的泛型的Class 獲取到Spring容器中該類型的所有bean”,因此我們就需要傳入ApplicationContext。那就需要實現ApplicationContextAware接口,下面的代碼來自網絡。
@Component public class SpringUtil implements ApplicationContextAware { private static ApplicationContext applicationContext; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { SpringUtil.applicationContext = applicationContext; } public static ApplicationContext getApplicationContext() { return applicationContext; } /** * 通過class獲取所有該類型的bean * @param clazz * @return */ public static MapgetBeans(Class> clazz) { return getApplicationContext().getBeansOfType(clazz); } }
3.重新實現動態代理類
@Slf4j public class DynamicProxyBeanFactory implements InvocationHandler { private String className; public DynamicProxyBeanFactory(String className) { this.className = className; @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //獲取實現className類的所有bean MapclassMap =SpringUtil.getBeans(Class.forName(className)); log.info("DynamicProxyBeanFactory className:{} impl class:{}",className,classMap); //這里先隨便返回一個 return classMap.get("deviceHandlerUrlImpl"); } public static T newMapperProxy(String typeName,Class mapperInterface) { ClassLoader classLoader = mapperInterface.getClassLoader(); Class>[] interfaces = new Class[]{mapperInterface}; DynamicProxyBeanFactory proxy = new DynamicProxyBeanFactory(typeName); return (T) Proxy.newProxyInstance(classLoader, interfaces, proxy); } }
調用:
Set> classes = getHandlerRouterClasses(); //獲取HandlerRouter的子接口的泛型Class 例如:DeviceHandlerRouter接口的DeviceHandler for (Class> clazz : classes) { //clazz 對應DeviceHandlerRouter.class //clazz 對應DeviceHandlerRouter.class //clazz 對應DeviceHandlerRouter.class Type[] types = clazz.getGenericInterfaces(); ParameterizedType type = (ParameterizedType) types[0]; //typeName對應DeviceHandlerRouter extends HandlerRouter 中的DeviceHandler.class //typeName對應DeviceHandlerRouter extends HandlerRouter 中的DeviceHandler.class //typeName對應DeviceHandlerRouter extends HandlerRouter 中的DeviceHandler.class String typeName = type.getActualTypeArguments()[0].getTypeName(); DeviceHandler deviceHandler = DynamicProxyBeanFactory.newMapperProxy(typeName,clazz); }
總結:這篇文章我們
1.通過反射獲取了HandlerRouter的所有子接口(這里是DeviceHandlerRouter),以及子接口對應的泛型。
2.重寫動態代理類,實現通過Spring的ApplicationContext查找出傳入的所有泛型的實現bean
下一篇:SpringBoot 動態代理|反射|注解|AOP 優化代碼(三)-注解
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://specialneedsforspecialkids.com/yun/75288.html
摘要:上一篇動態代理反射注解優化代碼二反射我們實現了通過反射完善找到目標類,然后通過動態代理提供默認實現,本篇我們將使用自定義注解來繼續優化。下一篇動態代理反射注解四動態代理對象注入到容器 上一篇SpringBoot 動態代理|反射|注解|AOP 優化代碼(二)-反射 我們實現了通過反射完善找到目標類,然后通過動態代理提供默認實現,本篇我們將使用自定義注解來繼續優化。 創建注解 1.創建枚舉...
摘要:生產環境由注冊中心,通過調用,其它環境直接通過直接通過調用。當然動態代理提供接口的默認實現只是演示,并沒有什么實際內容。下一篇動態代理反射注解優化代碼二反射 一、背景 在項目中需要調用外部接口,由于需要調用不同環境(生產、測試、開發)的相同接口(例如:向生、測試、開發環境的設備下發同一個APP)。 1.生產環境由SpringCloud注冊中心,通過Feign調用, 2.其它環境直接通過...
摘要:上一篇動態代理反射注解優化代碼三注解本篇我們將實現通過代理生成的對象注入到容器中。單元測試優化代碼待續參考文章 上一篇:SpringBoot 動態代理|反射|注解|AOP 優化代碼(三)-注解 本篇我們將實現通過代理生成的對象注入到spring容器中。首先需要實現BeanDefinitionRegistryPostProcessor, ApplicationContextAware兩個...
摘要:總結動態代理的相關原理已經講解完畢,接下來讓我們回答以下幾個思考題。 【干貨點】 此處是【好好面試】系列文的第12篇文章。文章目標主要是通過原理剖析的方式解答Aop動態代理的面試熱點問題,通過一步步提出問題和了解原理的方式,我們可以記得更深更牢,進而解決被面試官卡住喉嚨的情況。問題如下 SpringBoot默認代理類型是什么 為什么不用靜態代理 JDK動態代理原理 CGLIB動態代理...
閱讀 3430·2023-04-25 22:44
閱讀 935·2021-11-15 11:37
閱讀 1637·2019-08-30 15:55
閱讀 2650·2019-08-30 15:54
閱讀 1089·2019-08-30 13:45
閱讀 1437·2019-08-29 17:14
閱讀 1859·2019-08-29 13:50
閱讀 3416·2019-08-26 11:39