Spring中的Aware接口詳細(xì)解析
Aware接口介紹
Aware是一個(gè)具有標(biāo)識(shí)作用的超級(jí)接口,具體實(shí)現(xiàn)是有子接口去決定的,但是子接口至少要有一個(gè)帶一個(gè)參數(shù)的且返回是空的方法。實(shí)現(xiàn)該接口的bean是具有被spring 容器通知的能力的,而被通知的方式就是通過回調(diào)。也就是說:直接或間接實(shí)現(xiàn)了這個(gè)接口的類,都具有被spring容器通知的能力。
Aware翻譯過來是adj. 知道的,明白的,察覺到的,意識(shí)到的,所以這些接口從字面意思應(yīng)該是能感知到所有Aware前面的含義。 比如實(shí)現(xiàn)了ApplicationContextAware接口的類,能夠獲取到ApplicationContext,實(shí)現(xiàn)了BeanFactoryAware接口的類,能夠獲取到BeanFactory對(duì)象。
package org.springframework.beans.factory; /** * A marker superinterface indicating that a bean is eligible to be notified by the * Spring container of a particular framework object through a callback-style method. * The actual method signature is determined by individual subinterfaces but should * typically consist of just one void-returning method that accepts a single argument. * * <p>Note that merely implementing {@link Aware} provides no default functionality. * Rather, processing must be done explicitly, for example in a * {@link org.springframework.beans.factory.config.BeanPostProcessor}. * Refer to {@link org.springframework.context.support.ApplicationContextAwareProcessor} * for an example of processing specific {@code *Aware} interface callbacks. * * @author Chris Beams * @author Juergen Hoeller * @since 3.1 */ public interface Aware { }
Aware常用子接口如下:
org.springframework.context.ApplicationContextAware
org.springframework.beans.factory.BeanFactoryAware
org.springframework.beans.factory.BeanClassLoaderAware
org.springframework.beans.factory.BeanNameAware
org.springframework.context.EnvironmentAware
org.springframework.context.ResourceLoaderAware
org.springframework.context.annotation.ImportAware
1.ApplicationContextAware
實(shí)現(xiàn)該接口的類可以獲取spring容器上下文信息 ApplicationContext , ApplicationContextAware接口源碼如下:
public interface ApplicationContextAware extends Aware { void setApplicationContext(ApplicationContext applicationContext) throws BeansException; }
下面demo可通過類的靜態(tài)方法調(diào)用來獲取applicationContext上下文信息:
@Component public class ApplicationContextStaticProvider implements ApplicationContextAware { private static ApplicationContext context; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { context = applicationContext; } public static ApplicationContext getApplicationContext() { return context; } /** * 通過beanName獲取Bean * * @param beanName * @return */ public static Object getBean(String beanName) { return getApplicationContext().getBean(beanName); } /** * 通過class獲取bean * * @param clazz * @param <T> * @return */ public static <T> T getBean(Class<T> clazz) { return getApplicationContext().getBean(clazz); } /** * 通過name,以及Clazz返回指定的bean * * @param beanName * @param clazz * @param <T> * @return */ public static <T> T getBean(String beanName, Class<T> clazz) { return getApplicationContext().getBean(beanName, clazz); } } @Service public class DemoBean { public void test(String str){ System.out.println(str); } } public class AnnotationConfigApplicationContextTests { @Test public void scanAndRefreshTestAware() { AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(); context.scan("org.springframework.context.annotation7"); context.refresh(); DemoBean demoBean= (DemoBean) ApplicationContextStaticProvider.getBean("demoBean"); demoBean.test("test applicationContextAware"); } }
調(diào)用過程分析
Spring在初始化Bean時(shí),如何回調(diào)ApplicationContextAware接口setApplicationContext方法呢?
我們來看如下調(diào)用順序:
1.AbstractApplicationContext#refresh() ;
2.AbstractApplicationContext#prepareBeanFactory() —> beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));
3.AbstractApplicationContext#finishBeanFactoryInitialization();
4.DefaultListableBeanFactory#preInstantiateSingletons()—>getBean(beanName);
5.AbstractAutowireCapableBeanFactory#doCreateBean();
6.AbstractAutowireCapableBeanFactory#initializeBean();填充bean屬性后,初始化bean。
7.AbstractAutowireCapableBeanFactory#applyBeanPostProcessorsBeforeInitialization()
8.執(zhí)行ApplicationContextAwareProcessor.postProcessBeforeInitialization()
9.ApplicationContextAwareProcessor#invokeAwareInterfaces()
private void invokeAwareInterfaces(Object bean) { if (bean instanceof Aware) { if (bean instanceof EnvironmentAware) { ((EnvironmentAware) bean).setEnvironment(this.applicationContext.getEnvironment()); } if (bean instanceof EmbeddedValueResolverAware) { ((EmbeddedValueResolverAware) bean).setEmbeddedValueResolver(this.embeddedValueResolver); } if (bean instanceof ResourceLoaderAware) { ((ResourceLoaderAware) bean).setResourceLoader(this.applicationContext); } if (bean instanceof ApplicationEventPublisherAware) { ((ApplicationEventPublisherAware) bean).setApplicationEventPublisher(this.applicationContext); } if (bean instanceof MessageSourceAware) { ((MessageSourceAware) bean).setMessageSource(this.applicationContext); } // 最終調(diào)用ApplicationContextAware.setApplicationContext()方法 if (bean instanceof ApplicationContextAware) { ((ApplicationContextAware) bean).setApplicationContext(this.applicationContext); } } }
第一步: AbstractApplicationContext#refresh()
第二步:給bean工廠配置ApplicationContextAware回調(diào)處理 AbstractApplicationContext#prepareBeanFactory() —> beanFactory.addBeanPostProcessor(new ApplicationContextAwareProcessor(this));
第三步:初始化bean的過程(實(shí)例化(一般調(diào)用構(gòu)造方法實(shí)例化bean)、屬性注入、bean初始化),屬性注入后,bean初始化前,調(diào)用ApplicationContextAware實(shí)現(xiàn)類setApplicationContext()方法
2.BeanFactoryAware
Spring Ioc 中最頂層的父接口就是BeanFactory。實(shí)現(xiàn)這個(gè)BeanFactoryAware接口的子類可以獲取spring容器的BeanFactory 對(duì)象,進(jìn)而可以動(dòng)態(tài)的去操作 要在spring容器中注入的bean。
ApplicationContext接口是BeanFactory的子接口,所以繼承ApplicationContextAware的實(shí)現(xiàn)類拿到ApplicationContext 對(duì)象比實(shí)現(xiàn)BeanFactoryAware接口拿到BeanFactory 對(duì)象 可以獲取更多的信息。
BeanFactroy采用的是延遲加載形式來注入Bean的,即只有在使用到某個(gè)Bean時(shí)(調(diào)用getBean()),才對(duì)該Bean進(jìn)行加載實(shí)例化,這樣,我們就不能提前發(fā)現(xiàn)一些Spring的配置問題。而ApplicationContext則相反,它是在容器啟動(dòng)時(shí),一次性創(chuàng)建了所有的Bean(單例)。這樣,在容器啟動(dòng)時(shí),我們就可以發(fā)現(xiàn)Spring中存在的配置錯(cuò)誤。 相對(duì)于基本的BeanFactory,ApplicationContext 唯一的不足是占用內(nèi)存空間。當(dāng)應(yīng)用程序配置Bean較多時(shí),程序啟動(dòng)較慢。BeanFacotry延遲加載,如果Bean的某一個(gè)屬性沒有注入,BeanFacotry加載后,直至第一次使用調(diào)用getBean方法才會(huì)拋出異常;而ApplicationContext則在初始化自身時(shí)檢驗(yàn),這樣有利于檢查所依賴屬性是否注入;所以通常情況下我們選擇使用 ApplicationContext。應(yīng)用上下文則會(huì)在上下文啟動(dòng)后預(yù)載入所有的單實(shí)例Bean。通過預(yù)載入單實(shí)例bean ,確保當(dāng)你需要的時(shí)候,你就不用等待,因?yàn)樗鼈円呀?jīng)創(chuàng)建好了。
3.BeanNameAware
這個(gè)接口的含義就是讓實(shí)現(xiàn)類知道自己在spring容器中定義的beanName是什么,實(shí)際開發(fā)一般沒啥用。
4.BeanClassLoaderAware
獲取spring 容器的類加載器ClassLoader 對(duì)象; org.springframework.beans.factory.support.AbstractAutowireCapableBeanFactory#invokeAwareMethods
private void invokeAwareMethods(final String beanName, final Object bean) { if (bean instanceof Aware) { if (bean instanceof BeanNameAware) { ((BeanNameAware) bean).setBeanName(beanName); } if (bean instanceof BeanClassLoaderAware) { ClassLoader bcl = getBeanClassLoader(); if (bcl != null) { ((BeanClassLoaderAware) bean).setBeanClassLoader(bcl); } } if (bean instanceof BeanFactoryAware) { ((BeanFactoryAware) bean).setBeanFactory(AbstractAutowireCapableBeanFactory.this); } } }
正常使用的是: Thread.currentThread().getContextClassLoader();
5.EnvironmentAware
實(shí)現(xiàn)這個(gè)接口的類能獲取Environmet對(duì)象,進(jìn)而可以各種系統(tǒng)變量信息,也可以設(shè)置 變量的優(yōu)先級(jí)別等等。 通過Environment 對(duì)象可以獲取spring boot框架中的application.properties中定義的變量值。
6.ResourceLoaderAware
獲取資源加載器ResourceLoader 對(duì)象,可以獲得外部資源文件。
到此這篇關(guān)于Spring中的Aware接口詳細(xì)解析的文章就介紹到這了,更多相關(guān)Spring的Aware接口內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot接收參數(shù)所有方式總結(jié)
這篇文章主要介紹了SpringBoot接收參數(shù)所有方式總結(jié),文中通過代碼示例和圖文結(jié)合的方式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2024-07-07Spring下token過期時(shí)間分平臺(tái)(web和app)設(shè)置方法
本文詳細(xì)介紹了在Spring環(huán)境下,針對(duì)web端和APP端實(shí)現(xiàn)不同token過期時(shí)間的方法,通過整合SpringBoot、springSecurity和JWT框架,文章講解了登錄流程、JWT的基本組成以及token鑒權(quán)的核心步驟,需要的朋友可以參考下2024-10-10java反射校驗(yàn)參數(shù)是否是基礎(chǔ)類型步驟示例
這篇文章主要為大家介紹了java反射校驗(yàn)參數(shù)是否是基礎(chǔ)類型步驟示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12用java實(shí)現(xiàn)的獲取優(yōu)酷等視頻縮略圖的實(shí)現(xiàn)代碼
想獲取優(yōu)酷等視頻縮略圖,在網(wǎng)上沒有找到滿意的資料,參考了huangdijia的PHP版工具一些思路,寫了下面的JAVA版代碼。。其實(shí)也可以做成JS版的2013-05-05java遍歷讀取整個(gè)redis數(shù)據(jù)庫實(shí)例
這篇文章主要介紹了java遍歷讀取整個(gè)redis數(shù)據(jù)庫實(shí)例,使用支持正則表達(dá)式的key搜索方法jedis.keys(“*”)實(shí)現(xiàn),需要的朋友可以參考下2014-05-05深入解析Java多態(tài)進(jìn)階學(xué)習(xí)
java的動(dòng)態(tài)綁定機(jī)制非常重要。這篇文章將帶大家更深入的學(xué)習(xí)一下Java的多態(tài),文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Java有一定幫助,需要的可以參考一下2022-07-07Java list與set中contains()方法效率案例詳解
這篇文章主要介紹了Java list與set中contains()方法效率案例詳解,本篇文章通過簡要的案例,講解了該項(xiàng)技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-08-08