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

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/531200.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/531200.shtml
英文地址,請注明出處:http://en.pswp.cn/news/531200.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

使用pdfBox實現pdf轉圖片,解決中文方塊亂碼等問題

一、引入依賴 <dependency><groupId>org.apache.pdfbox</groupId><artifactId>fontbox</artifactId><version>2.0.13</version> </dependency> <dependency><groupId>org.apache.pdfbox</groupId><artif…

Spring異步調用原理及SpringAop攔截器鏈原理

一、Spring異步調用底層原理 開啟異步調用只需一個注解EnableAsync Target(ElementType.TYPE) Retention(RetentionPolicy.RUNTIME) Documented Import(AsyncConfigurationSelector.class) public interface EnableAsync {/*** Indicate the async annotation type to be detec…

線程池優化之充分利用線程池資源

一、前言 最近做了電子發票的需求&#xff0c;分省開票接口和發票下載接口都有一定的延遲。為了完成開票后自動將發票插入用戶微信卡包&#xff0c;目前的解決方案是利用線程池&#xff0c;將開票后插入卡包的任務&#xff08;輪詢分省發票接口&#xff0c;直到獲取到發票相關信…

Spring MVC源碼——Root WebApplicationContext

Spring MVC源碼——Root WebApplicationContext 打算開始讀一些框架的源碼,先拿 Spring MVC 練練手,歡迎點擊這里訪問我的源碼注釋, SpringMVC官方文檔一開始就給出了這樣的兩段示例: WebApplicationInitializer示例: public class MyWebApplicationInitializer implements Web…

Spring MVC源碼——Servlet WebApplicationContext

上一篇筆記(Spring MVC源碼——Root WebApplicationContext)中記錄了下 Root WebApplicationContext 的初始化代碼.這一篇來看 Servlet WebApplicationContext 的初始化代碼 DispatcherServlet 是另一個需要在 web.xml 中配置的類, Servlet WebApplicationContext 就由它來創建…

Springboot源碼——應用程序上下文分析

前兩篇(Spring MVC源碼——Root WebApplicationContext 和 Spring MVC源碼——Servlet WebApplicationContext)講述了springmvc項目創建上下文的過程&#xff0c;這一篇帶大家了解一下springboot項目創建上下文的過程。 SpringApplication引導類 SpringApplication類用于啟動或…

基于zookeeper實現分布式配置中心(一)

最近在學習zookeeper&#xff0c;發現zk真的是一個優秀的中間件。在分布式環境下&#xff0c;可以高效解決數據管理問題。在學習的過程中&#xff0c;要深入zk的工作原理&#xff0c;并根據其特性做一些簡單的分布式環境下數據管理工具。本文首先對zk的工作原理和相關概念做一下…

基于zookeeper實現分布式配置中心(二)

上一篇&#xff08;基于zookeeper實現分布式配置中心&#xff08;一&#xff09;&#xff09;講述了zookeeper相關概念和工作原理。接下來根據zookeeper的特性&#xff0c;簡單實現一個分布式配置中心。 配置中心的優勢 1、各環境配置集中管理。 2、配置更改&#xff0c;實時推…

Redis分布式鎖實戰

背景 目前開發過程中&#xff0c;按照公司規范&#xff0c;需要依賴框架中的緩存組件。不得不說&#xff0c;做組件的大牛對CRUD操作的封裝&#xff0c;連接池、緩存路由、緩存安全性的管控都處理的無可挑剔。但是有一個小問題&#xff0c;該組件沒有對分布式鎖做實現&#xff…

基于RobotFramework實現自動化測試

Java robotframework seleniumlibrary 使用Robot Framework Maven Plugin&#xff08;http://robotframework.org/MavenPlugin/&#xff09;執行自動化測試chromedriver下載&#xff1a; http://chromedriver.storage.googleapis.com/index.htmlchromedriver和chrome版本對應…

Springboot國際化信息(i18n)解析

國際化信息理解 國際化信息也稱為本地化信息 。 Java 通過 java.util.Locale 類來表示本地化對象&#xff0c;它通過 “語言類型” 和 “國家/地區” 來創建一個確定的本地化對象 。舉個例子吧&#xff0c;比如在發送一個具體的請求的時候&#xff0c;在header中設置一個鍵值對…

看了就知道為什么別人C語言學習效率那么高了

談及C語言&#xff0c;我想C語言功能強大都應該知道、應用廣泛&#xff0c;一旦掌握了后&#xff0c;你就可以理直氣壯地對他人說“我是電腦高手&#xff01;”&#xff0c;而且以后若是再自學其他語言就顯得輕而易舉了。憂慮的是&#xff0c;C語言般博大精深&#xff0c;太難學…

C語言一看就能上手的干貨!你確定你不來看嗎?

本地環境設置 如果您想要設置 C 語言環境&#xff0c;您需要確保電腦上有以下兩款可用的軟件&#xff0c;文本編輯器和 C 編譯器。 文本編輯器 這將用于輸入您的程序。文本編輯器包括 Windows Notepad、OS Edit command、Brief、Epsilon、EMACS 和 vim/vi。文本編輯器的名稱…

C語言爆炸干貨,小白你還不來看看嘛!

①&#xff1a;數據類型 int(整型)&#xff0c;short int(短整型)&#xff0c;long int(長整型)&#xff0c; char(字符型)&#xff0c;float&#xff08;單精度浮點型&#xff09; double&#xff08;雙精度浮點型&#xff09; C語言編程交流群815393895 ②&#xff1a;邏…

10萬碼農五年的C語言筆記!你現在知道別人為什么這么優秀了嗎?

c語言對許多同學來說確實是一門比較難學的課程&#xff0c;不僅抽象&#xff0c;而且繁瑣&#xff0c;但這又是一門不得不學的課程。前兩節可能還有興致聽一聽&#xff0c;然而&#xff0c;再過幾節課就是一臉蒙比。憑空要想出一道題的算法和程序&#xff0c;根本無從下手。 所…

C語言從來都沒有過時,你大爺終究是你大爺

直到今天&#xff0c;有人在喊C語言過時的語言&#xff0c;還有什么值得學習的&#xff0c;現在看Python&#xff0c;PHP等語言現在都很容易用&#xff0c;誰還在學習老C語言&#xff0c;其實這是真的嗎&#xff1f;作者下載了兩種語言的源代碼作為下載器。由于空間的限制&…

C語言超級瑪麗菜單模塊源碼

C語言是面向過程的&#xff0c;而C&#xff0b;&#xff0b;是面向對象的 C和C的區別&#xff1a; C是一個結構化語言&#xff0c;它的重點在于算法和…

C語言使用函數必須知道的3點注意事項!

C語言是面向過程的&#xff0c;而C&#xff0b;&#xff0b;是面向對象的 C和C的區別&#xff1a; C是一個結構化語言&#xff0c;它的重點在于算法和數據結構。C程序的設計首要考慮的是如何通過一個過程&#xff0c;對輸入&#xff08;或環境條件&#xff09;進行運算處理得…

C語言/C++編程學習:C語言環境設置!

C語言是面向過程的&#xff0c;而C&#xff0b;&#xff0b;是面向對象的 C和C的區別&#xff1a; C是一個結構化語言&#xff0c;它的重點在于算法和數據結構。C程序的設計首要考慮的是如何通過一個過程&#xff0c;對輸入&#xff08;或環境條件&#xff09;進行運算處理得…

C語言指針原來也可以這么的通俗易懂!

C語言是面向過程的&#xff0c;而C&#xff0b;&#xff0b;是面向對象的 C和C的區別&#xff1a; C是一個結構化語言&#xff0c;它的重點在于算法和數據結構。C程序的設計首要考慮的是如何通過一個過程&#xff0c;對輸入&#xff08;或環境條件&#xff09;進行運算處理得…