成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Spring奇技淫巧之擴展點的應用

開發(fā) 架構
bean生命周期的最后一個擴展點,該方法用于執(zhí)行一些bean銷毀前的準備工作,比如將當前bean持有的一些資源釋放掉。

[[392369]]

本文轉載自微信公眾號「月伴飛魚」,作者日常加油站。轉載本文請聯系月伴飛魚公眾號。  

最近在看公司項目和中間件的時候,看到一些Spring擴展點的使用,寫篇文章學習下,對大家之后看源碼都有幫助

「首先先介紹下Bean的生命周期」

我們知道Bean的生命周期分為幾個主干流程

  • Bean(單例非懶加載)的實例化階段
  • Bean的屬性注入階段
  • Bean的初始化階段
  • Bean的銷毀階段

下面是整個Spring容器的啟動流程,可以看到除了上述幾個主干流程外,Spring還提供了很多擴展點

下面詳細介紹下Spring的常見的擴展點

Spring常見擴展點

「BeanFactoryPostProcessor#postProcessBeanFactory」

有時候整個項目工程中bean的數量有上百個,而大部分單測依賴都是整個工程的xml,導致單測執(zhí)行時需要很長時間(大部分時間耗費在xml中數百個單例非懶加載的bean的實例化及初始化過程)

解決方法:利用Spring提供的擴展點將xml中的bean設置為懶加載模式,省去了Bean的實例化與初始化時間

  1. public class LazyBeanFactoryProcessor implements BeanFactoryPostProcessor { 
  2.     @Override 
  3.     public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException { 
  4.         DefaultListableBeanFactory fac = (DefaultListableBeanFactory) beanFactory; 
  5.         Map<String, AbstractBeanDefinition> map = (Map<String, AbstractBeanDefinition>) ReflectionTestUtils.getField(fac, "beanDefinitionMap"); 
  6.         for (Map.Entry<String, AbstractBeanDefinition> entry : map.entrySet()) { 
  7.             //設置為懶加載 
  8.             entry.getValue().setLazyInit(true); 
  9.         } 
  10.     } 

「InstantiationAwareBeanPostProcessor#postProcessPropertyValues」

非常規(guī)的配置項比如

  1. <context:component-scan base-package="com.zhou" /> 

Spring提供了與之對應的特殊解析器

正是通過這些特殊的解析器才使得對應的配置項能夠生效

而針對這個特殊配置的解析器為 ComponentScanBeanDefinitionParser

在這個解析器的解析方法中,注冊了很多特殊的Bean

  1. public BeanDefinition parse(Element element, ParserContext parserContext) { 
  2.   //... 
  3.   registerComponents(parserContext.getReaderContext(), beanDefinitions, element); 
  4.     //... 
  5.   return null
  1. public static Set<BeanDefinitionHolder> registerAnnotationConfigProcessors( 
  2.    BeanDefinitionRegistry registry, Object source) { 
  3.  
  4.   Set<BeanDefinitionHolder> beanDefs = new LinkedHashSet<BeanDefinitionHolder>(4); 
  5.   //... 
  6.     //@Autowire 
  7.   if (!registry.containsBeanDefinition(AUTOWIRED_ANNOTATION_PROCESSOR_BEAN_NAME)) { 
  8.    RootBeanDefinition def = new RootBeanDefinition(AutowiredAnnotationBeanPostProcessor.class); 
  9.    def.setSource(source); 
  10.    beanDefs.add(registerPostProcessor(registry, def, AUTOWIRED_ANNOTATION_PROCESSOR_BEAN_NAME)); 
  11.   } 
  12.  
  13.   // Check for JSR-250 support, and if present add the CommonAnnotationBeanPostProcessor. 
  14.    //@Resource 
  15.   if (jsr250Present && !registry.containsBeanDefinition(COMMON_ANNOTATION_PROCESSOR_BEAN_NAME)) { 
  16.       //特殊的Bean 
  17.    RootBeanDefinition def = new RootBeanDefinition(CommonAnnotationBeanPostProcessor.class); 
  18.    def.setSource(source); 
  19.    beanDefs.add(registerPostProcessor(registry, def, COMMON_ANNOTATION_PROCESSOR_BEAN_NAME)); 
  20.   } 
  21.   //... 
  22.   return beanDefs; 
  23.  } 

以@Resource為例,看看這個特殊的bean做了什么

  1. public class CommonAnnotationBeanPostProcessor extends InitDestroyAnnotationBeanPostProcessor 
  2.   implements InstantiationAwareBeanPostProcessor, BeanFactoryAware, Serializable { 
  3.       
  4.       public PropertyValues postProcessPropertyValues(PropertyValues pvs, PropertyDescriptor[] pds,  
  5.       Object bean, String beanName) throws BeansException { 
  6.           InjectionMetadata metadata = findResourceMetadata(beanName, bean.getClass()); 
  7.           try { 
  8.             //屬性注入 
  9.             metadata.inject(bean, beanName, pvs); 
  10.           } 
  11.           catch (Throwable ex) { 
  12.             throw new BeanCreationException(beanName, "Injection of resource dependencies failed", ex); 
  13.           } 
  14.           return pvs; 
  15.     } 
  16.      

我們看到在postProcessPropertyValues方法中,進行了屬性注入

「invokeAware」

實現BeanFactoryAware接口的類,會由容器執(zhí)行setBeanFactory方法將當前的容器BeanFactory注入到類中

  1. @Bean 
  2. class BeanFactoryHolder implements BeanFactoryAware{ 
  3.     
  4.     private static BeanFactory beanFactory; 
  5.      
  6.     public void setBeanFactory(BeanFactory beanFactory) throws BeansException { 
  7.         this.beanFactory = beanFactory; 
  8.     } 

「BeanPostProcessor#postProcessBeforeInitialization」

實現ApplicationContextAware接口的類,會由容器執(zhí)行setApplicationContext方法將當前的容器applicationContext注入到類中

  1. @Bean 
  2. class ApplicationContextAwareProcessor implements BeanPostProcessor { 
  3.  
  4.     private final ConfigurableApplicationContext applicationContext; 
  5.  
  6.     public ApplicationContextAwareProcessor(ConfigurableApplicationContext applicationContext) { 
  7.       this.applicationContext = applicationContext; 
  8.     } 
  9.  
  10.     @Override 
  11.     public Object postProcessBeforeInitialization(final Object bean, String beanName) throws BeansException { 
  12.       //... 
  13.       invokeAwareInterfaces(bean); 
  14.       return bean; 
  15.     } 
  16.  
  17.     private void invokeAwareInterfaces(Object bean) { 
  18.         if (bean instanceof ApplicationContextAware) { 
  19.           ((ApplicationContextAware) bean).setApplicationContext(this.applicationContext); 
  20.         } 
  21.     } 

我們看到是在BeanPostProcessor的postProcessBeforeInitialization中進行了setApplicationContext方法的調用

  1. class ApplicationContextHolder implements ApplicationContextAware{ 
  2.     
  3.     private static ApplicationContext applicationContext; 
  4.      
  5.     public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { 
  6.         this.applicationContext = applicationContext; 
  7.     } 

「afterPropertySet()和init-method」

目前很多Java中間件都是基本Spring Framework搭建的,而這些中間件經常把入口放到afterPropertySet或者自定義的init中

「BeanPostProcessor#postProcessAfterInitialization」

熟悉aop的同學應該知道,aop底層是通過動態(tài)代理實現的

當配置了時候,默認開啟aop功能,相應地調用方需要被aop織入的對象也需要替換為動態(tài)代理對象

不知道大家有沒有思考過動態(tài)代理是如何「在調用方無感知情況下替換原始對象」的?

根據上文的講解,我們知道:

  1. <aop:aspectj-autoproxy/> 

Spring也提供了特殊的解析器,和其他的解析器類似,在核心的parse方法中注冊了特殊的bean

這里是一個BeanPostProcessor類型的bean

  1. class AspectJAutoProxyBeanDefinitionParser implements BeanDefinitionParser { 
  2.  @Override 
  3.  public BeanDefinition parse(Element element, ParserContext parserContext) { 
  4.     //注冊特殊的bean 
  5.   AopNamespaceUtils.registerAspectJAnnotationAutoProxyCreatorIfNecessary(parserContext, element); 
  6.   extendBeanDefinition(element, parserContext); 
  7.   return null
  8.     } 

將于當前bean對應的動態(tài)代理對象返回即可,該過程對調用方全部透明

  1. public class AnnotationAwareAspectJAutoProxyCreator extends AspectJAwareAdvisorAutoProxyCreator { 
  2.   public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException { 
  3.         if (bean != null) { 
  4.           Object cacheKey = getCacheKey(bean.getClass(), beanName); 
  5.           if (!this.earlyProxyReferences.containsKey(cacheKey)) { 
  6.             //如果該類需要被代理,返回動態(tài)代理對象;反之,返回原對象 
  7.             return wrapIfNecessary(bean, beanName, cacheKey); 
  8.           } 
  9.         } 
  10.         return bean; 
  11.  } 

正是利用Spring的這個擴展點實現了動態(tài)代理對象的替換

「destroy()和destroy-method」

bean生命周期的最后一個擴展點,該方法用于執(zhí)行一些bean銷毀前的準備工作,比如將當前bean持有的一些資源釋放掉

 

責任編輯:武曉燕 來源: 月伴飛魚
相關推薦

2020-05-20 19:38:11

前端js調試工具

2017-08-18 13:30:01

前端CSS布局奇技

2022-01-07 14:50:46

VS CodeLinux代碼

2023-06-26 08:05:36

2017-10-24 13:42:55

流氓App安卓Google

2021-03-30 07:47:46

SVG 濾鏡 CSS技巧

2020-11-26 11:45:31

Python繪圖代碼

2022-09-30 12:55:14

Linux筆記

2022-04-21 15:00:53

LinuxShell

2023-09-19 08:03:50

rebase?merge

2015-04-13 13:21:45

JavaScript JavaScript

2021-02-25 09:19:11

LinuxAppimage命令

2021-05-18 13:05:31

LinuxRust復用器

2021-06-07 12:20:14

LinuxASCII命令

2019-04-25 13:10:04

Java 8Stream API編程語言

2022-04-24 16:00:15

LinuxLinux命令ls命令

2024-11-28 09:21:00

Python字符串代碼

2021-05-07 13:56:13

Linux器監(jiān)視服務器

2021-05-31 11:45:37

LinuxRustShell

2023-09-28 08:49:41

springBean
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产色视频网站 | 亚洲一级毛片 | 精品一区二区三区四区外站 | 国产福利91精品一区二区三区 | 欧美一区二 | 欧美精品99 | 欧美成人影院在线 | 日韩视频在线观看一区二区 | 欧美精品一区二区在线观看 | 欧美高清视频在线观看 | 中文字幕免费视频 | 午夜寂寞网站 | 欧美成人h版在线观看 | 日韩欧美精品一区 | 中文字幕黄色大片 | 国产.com | 狠狠色香婷婷久久亚洲精品 | 久久久久亚洲精品中文字幕 | 九九热精品在线 | 日日摸日日碰夜夜爽亚洲精品蜜乳 | 91久久久久久 | 97精品超碰一区二区三区 | 麻豆精品国产91久久久久久 | 日韩欧美一级精品久久 | 久久99精品久久久久久国产越南 | 极情综合网 | 精品一区二区三区在线观看 | 亚洲视频中文字幕 | 日韩欧美在线视频 | 亚洲一区欧美 | 国产乱一区二区三区视频 | 午夜精品久久久久久久星辰影院 | 99riav3国产精品视频 | 色呦呦在线 | 亚洲国产高清高潮精品美女 | 亚洲一区二区在线视频 | 国产成人精品一区二区三区四区 | 亚洲在线电影 | 免费在线看黄 | 精品欧美激情精品一区 | 亚洲综合热|