自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

@PostConstruct注解是Spring提供的?今天講點不一樣的

開發(fā) 架構
JSR-250主要圍繞著“資源”的使用預定義了一些注解(Annotation),這里的“資源”可以理解為一個Class類的實例、一個JavaBean、或者一個Spring中的Bean。

[[402828]]

本文轉(zhuǎn)載自微信公眾號「程序新視界」,作者二師兄。轉(zhuǎn)載本文請聯(lián)系程序新視界公眾號。

前言

我們在講《Spring的Lifecycle》時提到,在Spring的使用中可以通過Lifecycle接口實現(xiàn)一些基于Spring容器生命周期邏輯。與此對照的就是通過@PostConstruct和@PreDestroy在Bean初始化或銷毀時執(zhí)行一些操作。

很明顯Spring的Lifecycle是基于容器的生命周期來處理邏輯,而@PostConstruct和@PreDestroy是基于Bean的生命周期來處理業(yè)務邏輯。

這里很多朋友就產(chǎn)生了一個誤解,以為@PostConstruct注解也是Spring提供的。其實不然,它是Java自帶的注解,下面我們就從頭來聊聊@PostConstruct注解。

JSR-250規(guī)范

在了解@PostConstruct注解之前,我們先來科普一個概念:JSR-250規(guī)范。

JSR-250主要圍繞著“資源”的使用預定義了一些注解(Annotation),這里的“資源”可以理解為一個Class類的實例、一個JavaBean、或者一個Spring中的Bean。

JSR-250相關的注解全部在javax.annotation和javax.annotation.security包中,包括:資源定義和權限控制。像我們經(jīng)常用到的@Resource、@PostConstruct、@PreDestroy、@Generated等都屬于這個規(guī)范中定義的注解。

該規(guī)范并沒有提供具體的實現(xiàn)方式,僅僅是提供了指導性的文檔和幾個注解,由具體的框架去實現(xiàn)。

也就是說,@PostConstruct注解并不是Spring提供的注解,只不過Spring按照JSR-250規(guī)范實現(xiàn)了規(guī)范中對@PostConstruct的約定。而別的框架,或者你自己寫一個框架,同樣可以按照約定進行實現(xiàn)。

@PostConstruct的約定

@PostConstruct和@PreDestroy是在Java EE 5引入的,位于javax.annotation包下,也就是java拓展包定義的注解。其中,javax中的x就是extension的意思。Java最初的設計者認為,這些功能并不是Java核心API,因此就放到了擴展包中,誰用誰實現(xiàn),按照約定就行。

下面直接看看該類上的注解說明:

“PostConstruct注釋用于在依賴關系注入完成之后需要執(zhí)行的方法上,以執(zhí)行任何初始化。此方法必須在將類放入服務之前調(diào)用。支持依賴關系注入的所有類都必須支持此注釋。即使類沒有請求注入任何資源,用PostConstruct注釋的方法也必須被調(diào)用。只有一個方法可以用此注釋進行注釋。”

“應用PostConstruct注釋的方法必須遵守以下所有標準:

  • 該方法不得有任何參數(shù),除非是在EJB攔截器(interceptor)的情況下,它將帶有一個InvocationContext對象;
  • 該方法的返回類型必須為void;
  • 該方法不得拋出已檢查異常;
  • 應用PostConstruct的方法可以是public、protected、package private或private;
  • 除了應用程序客戶端之外,該方法不能是static;
  • 該方法可以是final;
  • 如果該方法拋出未檢查異常,那么不得將類放入服務中,除非是能夠處理異常并可從中恢復的EJB。

除了上述約定,如果用在Servlet容器當中,還有有一定的處理時機。

@PostConstruct的執(zhí)行時機

下面所講的@PostConstruct的執(zhí)行時機是基于Spring的實現(xiàn)來講的。被@PostConstruct修飾的方法會在服務器加載Servlet時運行,并且只會被執(zhí)行一次。PostConstruct在構造函數(shù)之后執(zhí)行,init()方法之前執(zhí)行。

對應的流程圖如下:

實例演示

理解了上面的基本概念,就先來看一個實例演示吧,使用起來非常簡單。

基于Java 8的Spring Boot項目中添加如下類:

  1. @Service 
  2. public class OrderService { 
  3.  
  4.     public OrderService(){ 
  5.         System.out.println("OrderService構造方法被執(zhí)行..."); 
  6.     } 
  7.  
  8.     @PostConstruct 
  9.     private void init() { 
  10.         System.out.println("PostConstruct注解方法被調(diào)用"); 
  11.     } 
  12.  
  13.     @PreDestroy 
  14.     private void shutdown() { 
  15.         System.out.println("PreDestroy注解方法被調(diào)用"); 
  16.     } 
  17.  

啟動Spring Boot項目,控制臺打印日志如下:

  1. OrderService構造方法被執(zhí)行... 
  2. PostConstruct注解方法被調(diào)用 

當關閉服務時,會打印:

  1. PreDestroy注解方法被調(diào)用 

通過實例,基本印證了上述說的理論。

Java9的以后的移除

在Java 8中我們可以直接使用對應的注解即可,但到Java 9及以后,J2EE棄用了@PostConstruct和@PreDestroy這兩個注解,并計劃在Java 11中將其刪除。

針對這種情況,我們有兩種解決方案:第一添加額外的依賴;第二,換用其他的方式。

第一種方案針對的是,你非要使用這個注解,或者說你的項目暫時沒辦法棄用這兩個注解。那么,可以手動添加依賴:

  1. <dependency> 
  2.     <groupId>javax.annotation</groupId> 
  3.     <artifactId>javax.annotation-api</artifactId> 
  4.     <version>1.3.2</version> 
  5. </dependency> 

也就是說,雖然移除了,但是你把它們的依賴添加上,依舊還是可以用的。但此時也給我們提了一個醒兒,在項目中盡量別用這兩個注解了,Java 11都計劃將其移除了。

此時,如果你使用的是Spring的項目,則可考慮另外一種方式,基于Spring的InitializingBean和DisposableBean接口來實現(xiàn)同樣的功能:

  1. @Service 
  2. public class PaymentService implements InitializingBean, DisposableBean { 
  3.  
  4.     public PaymentService(){ 
  5.         System.out.println("PaymentService構造方法被執(zhí)行..."); 
  6.     } 
  7.  
  8.     @Override 
  9.     public void destroy() throws Exception { 
  10.         System.out.println("destroy方法被調(diào)用"); 
  11.     } 
  12.  
  13.     @Override 
  14.     public void afterPropertiesSet() throws Exception { 
  15.         System.out.println("afterPropertiesSet方法被調(diào)用"); 
  16.     } 

啟動項目,打印日志如下:

  1. PaymentService構造方法被執(zhí)行... 
  2. afterPropertiesSet方法被調(diào)用 

停止項目,打印如下信息:

  1. destroy方法被調(diào)用 

也就是說在Spring的生態(tài)中,我們已經(jīng)有替代方案可實現(xiàn)了,而且是比較推薦的方式。

其實Spring并沒有遵守約定

在上面的約定中我們講到一個類中“只有一個方法可以用此注釋進行注釋”,在OrderService中再添加一個@PostConstruct注解的方法試試:

  1. @Service 
  2. public class OrderService { 
  3.  
  4.     public OrderService(){ 
  5.         System.out.println("OrderService構造方法被執(zhí)行..."); 
  6.     } 
  7.  
  8.     @PostConstruct 
  9.     private void init() { 
  10.         System.out.println("PostConstruct注解方法被調(diào)用"); 
  11.     } 
  12.  
  13.     @PostConstruct 
  14.     private void init1() { 
  15.         System.out.println("PostConstruct init1 注解方法被調(diào)用"); 
  16.     } 
  17.  
  18.     @PreDestroy 
  19.     private void shutdown() { 
  20.         System.out.println("PreDestroy注解方法被調(diào)用"); 
  21.     } 
  22.  

啟動程序,打印日志:

  1. OrderService構造方法被執(zhí)行... 
  2. PostConstruct init1 注解方法被調(diào)用 
  3. PostConstruct注解方法被調(diào)用 

不但沒報錯,而且兩個方法還都執(zhí)行了。這說明什么?這說明約定有時候就是用來被打破的,記住這一特殊情況就好。

Spring中的實現(xiàn)原理

以上是對@PostConstruct的簡單介紹,下面會從Spring源碼層面簡單分析一下實現(xiàn)原理。

我們先來看一個Spring的接口BeanPostProcessor:

  1. public interface BeanPostProcessor { 
  2.   
  3.     // 任何Bean實例化,并且Bean已經(jīng)populated(填充屬性) 就會回調(diào)這個方法 
  4.     Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException; 
  5.   
  6.     // 任何Bean實例化,并且Bean已經(jīng)populated(填充屬性) 就會回調(diào)這個方法 
  7.     Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException; 

BeanPostProcessor是Spring IOC容器給我們提供的一個擴展接口,它兩個回調(diào)方法。當一個BeanPostProcessor的實現(xiàn)類注冊到Spring IOC容器后,對于該Spring IOC容器所創(chuàng)建的每個bean實例在初始化方法(如afterPropertiesSet和任意已聲明的init方法)調(diào)用前,將會調(diào)用BeanPostProcessor中的postProcessBeforeInitialization方法,而在bean實例初始化方法調(diào)用完成后,則會調(diào)用BeanPostProcessor中的postProcessAfterInitialization方法,整個調(diào)用順序可以簡單示意如下:

  1. --> Spring IOC容器實例化Bean 
  2. --> 調(diào)用BeanPostProcessor的postProcessBeforeInitialization方法 
  3. --> 調(diào)用bean實例的初始化方法 
  4. --> 調(diào)用BeanPostProcessor的postProcessAfterInitialization方法 

而BeanPostProcessor有個實現(xiàn)類CommonAnnotationBeanPostProcessor,就是專門處理@PostConstruct和@PreDestroy注解。其中CommonAnnotationBeanPostProcessor的父類InitDestroyAnnotationBeanPostProcessor中,對應的調(diào)用邏輯如下:

  1. InitDestroyAnnotationBeanPostProcessor.postProcessBeforeInitialization() 
  2.     InitDestroyAnnotationBeanPostProcessor.findLifecycleMetadata() 
  3.         // 組裝生命周期元數(shù)據(jù) 
  4.         InitDestroyAnnotationBeanPostProcessor.buildLifecycleMetadata() 
  5.             // 查找@PostConstruct注釋的方法 
  6.             InitDestroyAnnotationBeanPostProcessor.initAnnotationType 
  7.             // 查找@PreDestroy注釋方法 
  8.             InitDestroyAnnotationBeanPostProcessor.destroyAnnotationType 
  9.  // 反射調(diào)用           
  10.  metadata.invokeInitMethods(bean, beanName); 

關于業(yè)務邏輯的處理細節(jié),這里就不再逐一展示,大家感興趣的話可以跟蹤一下源代碼。

小結

本篇文章我們需要留意幾點:第一,Spring只是實現(xiàn)了Java中對@PostConstruct注解定義的規(guī)范;第二,該注解在Java 9逐步開始廢棄,不建議再使用;第三,可采用Spring的InitializingBean和DisposableBean來替代對應的功能。

 

責任編輯:武曉燕 來源: 程序新視界
相關推薦

2012-03-07 17:24:10

戴爾咨詢

2012-12-20 10:17:32

IT運維

2017-05-25 15:02:46

聯(lián)宇益通SD-WAN

2015-10-19 12:33:01

華三/新IT

2016-05-09 18:40:26

VIP客戶緝拿

2021-02-01 06:10:02

springaop機制開發(fā)

2018-05-09 15:42:24

新零售

2009-02-04 15:43:45

敏捷開發(fā)PHPFleaPHP

2009-12-01 16:42:27

Gentoo Linu

2011-02-28 10:38:13

Windows 8

2009-06-12 15:26:02

2016-03-24 18:51:40

2015-08-25 09:52:36

云計算云計算產(chǎn)業(yè)云計算政策

2019-01-03 14:39:08

Oracle甲骨文ORACLE

2022-05-05 21:47:32

Linuxls 命令

2013-01-11 18:10:56

軟件

2015-08-04 14:49:54

Discover

2009-07-07 10:44:14

多態(tài)

2009-11-26 13:16:25

Open Suse

2018-06-26 11:10:54

UbuntuSUSE紅帽
點贊
收藏

51CTO技術棧公眾號