日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 前端技术 > javascript >内容正文

javascript

Spring BPP中优雅的创建动态代理Bean

發布時間:2025/3/8 javascript 23 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Spring BPP中优雅的创建动态代理Bean 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

一、前言

  本文章所講并沒有基于Aspectj,而是直接通過Cglib以及ProxyFactoryBean去創建代理Bean。通過下面的例子,可以看出Cglib方式創建的代理Bean和ProxyFactoryBean創建的代理Bean的區別。

二、基本測試代碼

  測試實體類,在BPP中創建BppTestDepBean類型的代理Bean。

@Component public static class BppTestBean {@Autowiredprivate BppTestDepBean depBean;public void test1() {depBean.testDep();}public void test2() {depBean.testDep();}@TestMethodpublic void test3() {depBean.testDep();} }@Component public static class BppTestDepBean {public void testDep() {System.out.println("HEHE");} }@Target(ElementType.METHOD) @Retention(RetentionPolicy.RUNTIME) @Documented public @interface TestMethod { }

  測試類

@RunWith(SpringRunner.class) @SpringBootTest public class BppTest {@Autowiredprivate BppTestBean bppTestBean;@Testpublic void test() {bppTestBean.test1();bppTestBean.test2();bppTestBean.test3();} }

三、使用Cglib創建代理Bean

public class ProxyBpp1 implements BeanPostProcessor {private static final Logger LOGGER = LoggerFactory.getLogger(ProxyBpp1.class);@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {if (bean instanceof BppTestBean) {Enhancer enhancer = new Enhancer();enhancer.setSuperclass(bean.getClass());//標識Spring-generated proxiesenhancer.setInterfaces(new Class[]{SpringProxy.class});//設置增強enhancer.setCallback((MethodInterceptor) (target, method, args, methodProxy) -> {if ("test1".equals(method.getName())) {LOGGER.info("ProxyBpp1 開始執行...");Object result = methodProxy.invokeSuper(target, args);LOGGER.info("ProxyBpp1 結束執行...");return result;}return method.invoke(target, args);});return enhancer.create();}return bean;} }

  主要是代理?BppTestBean的test1方法。其實這種方式創建的代理Bean使用問題的,@Autowired字段沒有注入進來,所以會有出現NPE。methodProxy.invokeSuper(target, args),這一行代碼是有問題的,targe是代理類對象,而真實的對象是postProcessBeforeInitialization(Object bean, String beanName) 中的bean對象,此時bean對象@Autowired字段已經注入了。所以可以將methodProxy.invokeSuper(target, args) 修改為method.invoke(bean, args)解決無法注入@Autowired字段的問題。

四、使用ProxyFactoryBean創建代理Bean

public class ProxyBpp2 implements BeanPostProcessor {private static final Logger LOGGER = LoggerFactory.getLogger(ProxyBpp2.class);@Overridepublic Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {if (bean instanceof BppTestBean) {ProxyFactoryBean pfb = new ProxyFactoryBean();pfb.setTarget(bean);pfb.setAutodetectInterfaces(false);NameMatchMethodPointcutAdvisor advisor = new NameMatchMethodPointcutAdvisor();advisor.addMethodName("test1");advisor.setAdvice((MethodInterceptor) invocation -> {LOGGER.info("ProxyBpp2 開始執行...");Object result = invocation.getMethod().invoke(invocation.getThis(), invocation.getArguments());LOGGER.info("ProxyBpp2 結束執行...");return result;});pfb.addAdvisor(advisor);return pfb.getObject();}return bean;} }

   使用ProxyFactoryBean創建代理Bean的時候,一定要一個targe對象的。Advisor在切入的時候,會逐個執行Advice。invocation.getThis()就是在通過ProxyFactoryBean創建代理Bean的時候傳入的target對象。由于target對象就是postProcessBeforeInitialization(Object bean, String beanName) 中的bean對象,所以@Autowired字段也已經注入進來了。

五、@Autowired注解何時被處理

  想必大家都知道@Autowired字段的處理也是通過一個BPP,不過這個BPP比我們平常使用的要高級一些,它就是InstantiationAwareBeanPostProcessor。這個BPP可以實現Bean的創建、屬性的注入和解析(比如@Autowired、@Value、@Resource等等),大家可以參考一下CommonAnnotationBeanPostProcessor(處理JSR-250相關注解),AutowiredAnnotationBeanPostProcessor(處理@Autowired、@Value、@Inject相關注解)。

  InstantiationAwareBeanPostProcessor中有一個如下的方法,AutowiredAnnotationBeanPostProcessor就是覆蓋這個方法實現了帶有相關注解屬性的自動注入。

@Nullable default PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName)throws BeansException {return null; } @Override public PropertyValues postProcessProperties(PropertyValues pvs, Object bean, String beanName) {InjectionMetadata metadata = findAutowiringMetadata(beanName, bean.getClass(), pvs);try {metadata.inject(bean, beanName, pvs);}catch (BeanCreationException ex) {throw ex;}catch (Throwable ex) {throw new BeanCreationException(beanName, "Injection of autowired dependencies failed", ex);}return pvs; }

  InstantiationAwareBeanPostProcessor的postProcessProperties方法實在Spring?AbstractAutowireCapableBeanFactory的populateBean方法中被調用。在AbstractAutowireCapableBeanFactory的doCreateBan中有如下代碼。

// Initialize the bean instance. Object exposedObject = bean;# try {populateBean(beanName, mbd, instanceWrapper);exposedObject = initializeBean(beanName, exposedObject, mbd); }

  也就是先進行了Bean的屬性填充,然后進行Bean的初始化工作。initializeBean方法中主要做了四件事。

  1、invokeAwareMethods
  2、applyBeanPostProcessorsBeforeInitialization
  3、invokeInitMethods
  4、applyBeanPostProcessorsAfterInitialization

  其中2和4就是分別調用的普通的BPP中的postProcessBeforeInitialization方法和postProcessAfterInitialization方法。

  這就是為什么在BPP中創建代理Bean的時候,對應的目標Bean相關的@Autowired字段已經注入的原因了。

六、InstantiationAwareBeanPostProcessor方式創建動態代理Bean

  InstantiationAwareBeanPostProcessor接口中有個postProcessBeforeInstantiation方法,可以讓我們自己去實例化Bean。通過查看AbstractAutowireCapableBeanFactory,方法調用:createBean方法 ->?resolveBeforeInstantiation方法 ->?applyBeanPostProcessorsBeforeInstantiation方法 ->InstantiationAwareBeanPostProcessor#postProcessBeforeInstantiation方法,如果最終返回一個非null的實例,那么就不會再執行doCreateBean方法。這就意味著不會有Bean屬性的填充和初始化的流程了,但是可以借助AbstractAutowireCapableBeanFactory幫助我們實現。

public <T> T postProcess(T object) {if (object == null) {return null;}T result;try {// 使用容器autowireBeanFactory標準依賴注入方法autowireBean()處理 object對象的依賴注入this.autowireBeanFactory.autowireBean(object);// 使用容器autowireBeanFactory標準初始化方法initializeBean()初始化對象 objectresult = (T) this.autowireBeanFactory.initializeBean(object,object.toString());} catch (RuntimeException e) {Class<?> type = object.getClass();throw new RuntimeException("Could not postProcess " + object + " of type " + type, e);}return result; }

  上圖代碼,可以幫組我們實現非Spring容器Bean自動注入和初始化的功能。使用過Spring security同學都知道,內部也是用了這個方式解決對象中的屬性注入問題。如果你閱讀了Spring security的源碼,你會發現很多對象,比如WebSecurity、ProviderManager、各個安全Filter等,這些對象的創建并不是通過bean定義的形式被容器發現和注冊進入spring容器的,而是直接new出來的。Spring security提供的AutowireBeanFactoryObjectPostProcessor這個工具類可以使這些對象具有容器bean同樣的生命周期,也能注入相應的依賴,從而進入準備好被使用的狀態。

  使用Cglib在InstantiationAwareBeanPostProcessor 中創建動態代理Bean。

public class ProxyBpp3 implements InstantiationAwareBeanPostProcessor {private static final Logger LOGGER = LoggerFactory.getLogger(ProxyBpp3.class);private final AutowireCapableBeanFactory autowireBeanFactory;ProxyBpp3(AutowireCapableBeanFactory autowireBeanFactory) {this.autowireBeanFactory = autowireBeanFactory;}@Overridepublic Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) throws BeansException {if (beanClass.equals(BppConfig.BppTestBean.class)) {Enhancer enhancer = new Enhancer();enhancer.setSuperclass(beanClass);//標識Spring-generated proxiesenhancer.setInterfaces(new Class[]{SpringProxy.class});//設置增強enhancer.setCallback((MethodInterceptor) (target, method, args, methodProxy) -> {if ("test1".equals(method.getName())) {LOGGER.info("ProxyBpp3 開始執行...");Object result = methodProxy.invokeSuper(target, args);LOGGER.info("ProxyBpp3 結束執行...");return result;}return methodProxy.invokeSuper(target, args);});return this.postProcess(enhancer.create());}return null;}... }

  使用ProxyFactoryBean在InstantiationAwareBeanPostProcessor 中創建動態代理Bean。

public class ProxyBpp4 implements InstantiationAwareBeanPostProcessor {private static final Logger LOGGER = LoggerFactory.getLogger(ProxyBpp4.class);private final AutowireCapableBeanFactory autowireBeanFactory;ProxyBpp4(AutowireCapableBeanFactory autowireBeanFactory) {this.autowireBeanFactory = autowireBeanFactory;}@Overridepublic Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) throws BeansException {if (beanClass.equals(BppConfig.BppTestBean.class)) {ProxyFactoryBean pfb = new ProxyFactoryBean();pfb.setTarget(this.postProcess(BeanUtils.instantiateClass(beanClass)));pfb.setAutodetectInterfaces(false);NameMatchMethodPointcutAdvisor advisor = new NameMatchMethodPointcutAdvisor();advisor.addMethodName("test1");advisor.setAdvice((MethodInterceptor) invocation -> {LOGGER.info("ProxyBpp4 開始執行...");Object result = invocation.getMethod().invoke(invocation.getThis(), invocation.getArguments());LOGGER.info("ProxyBpp4 結束執行...");return result;});pfb.addAdvisor(advisor);return pfb.getObject();}return null;}... }

  上述向兩種方式,注意,實例化bean后主動通過postProcess方法借助AbstractAutowireCapableBeanFactory完成對象相關屬性的注入以及對象的初始化流程。

七、源碼分享

  點我查看源碼,如果有任何疑問請關注公眾號后進行咨詢。

轉載于:https://www.cnblogs.com/hujunzheng/p/10463798.html

總結

以上是生活随笔為你收集整理的Spring BPP中优雅的创建动态代理Bean的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 国产在线第一页 | 亚洲视频高清 | 欧美一级免费观看 | 这里有精品视频 | 亚洲成年人 | 欧美日韩第一区 | 亚洲av无码一区东京热久久 | 雨宫琴音一区二区三区 | 天天看夜夜| 婷婷中文在线 | 无码成人精品区一级毛片 | www.久久久久久久久久 | 黄色网址进入 | 国产中文一区二区三区 | 99热麻豆 | 亚洲伊人成人网 | 欧美日韩免费在线观看 | 色偷偷噜噜噜亚洲男人的天堂 | 色婷婷综合久久久久中文 | 朴麦妮原版视频高清资源 | 精品国偷自产在线 | 亚洲第一二三四区 | 都市激情校园春色亚洲 | 先锋av资源网 | 日本高清有码 | 黄色91 | 免费成人美女在线观看 | 亚洲精品在线中文字幕 | a级黄色网 | 欧美一级黑人 | 午夜精品在线播放 | 欧美精品一二 | 色黄大色黄女片免费中国 | 亚洲欧美另类中文字幕 | 成人拍拍 | 久色视频 | 2019毛片 | 猛男被粗大男男1069 | 伊人77| 国产美女白浆 | 亚洲欧美激情图片 | 久久黄色小说 | 肉色超薄丝袜脚交69xx | 黄色大片a级 | 国产精品探花一区二区在线观看 | 亚洲区 欧美区 | 欧美色欧美色 | 国产老女人乱淫免费 | 夜av| 国产成人三级在线 | 96毛片| 伊人网在线视频观看 | 日韩视频在线观看 | 欧美综合在线观看 | 国产免费av观看 | 亚洲视频在线观看 | 亚洲成人中文字幕在线 | 日韩射吧 | 国产免费不卡av | 午夜精品久久久久久久96蜜桃 | 亚洲资源站 | 人成在线视频 | 成人自拍在线 | 日韩av网站在线观看 | 超碰超碰超碰超碰 | 成人av在线看 | 500福利视频导航 | 国产手机看片 | 色视频一区 | 国产一级黄色大片 | xxxx黄色 | 97免费超碰 | 午夜久久久久 | 亚洲人在线播放 | av在线免费不卡 | 国产精品91久久 | 欧美乱大交xxxxx潮喷l头像 | 国产盗摄av | 久久艹精品视频 | 亚洲精品国产精品乱码在线观看 | 狼人伊人干 | 国产视频一区二区在线观看 | 久艹伊人 | 亚洲精品第一 | 日本东京热一区二区 | 中文字幕在线观看视频免费 | 亚洲男女在线观看 | 在线观看日韩视频 | 精产国品一二三区 | 97超碰人人在线 | 国产一级特黄毛片 | 在线观看国产麻豆 | 秋霞影院午夜老牛影院 | 成人黄色在线观看视频 | 性欧美成人播放77777 | 免费观看亚洲视频 | 1000部拍拍拍18勿入免费视频 | 岛国精品一区二区 | 亚洲免费三级 |