Spring容器注冊組件實現過程解析
1、@Configuration&@Bean給容器中注冊組件
@Configuration及@Bean的使用參考如下代碼:
package com.atguigu.config; import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.ComponentScan;import org.springframework.context.annotation.Configuration;import org.springframework.context.annotation.FilterType;import org.springframework.context.annotation.ComponentScan.Filter;import org.springframework.context.annotation.ComponentScans; import com.atguigu.bean.Person; //配置類==配置文件@Configuration //告訴Spring這是一個配置類 @ComponentScans( value = {@ComponentScan(value='com.atguigu',includeFilters = {/* @Filter(type=FilterType.ANNOTATION,classes={Controller.class}), @Filter(type=FilterType.ASSIGNABLE_TYPE,classes={BookService.class}),*/ @Filter(type=FilterType.CUSTOM,classes={MyTypeFilter.class})},useDefaultFilters = false) } )//@ComponentScan value:指定要掃描的包//excludeFilters = Filter[] :指定掃描的時候按照什么規則排除那些組件//includeFilters = Filter[] :指定掃描的時候只需要包含哪些組件//FilterType.ANNOTATION:按照注解//FilterType.ASSIGNABLE_TYPE:按照給定的類型;//FilterType.ASPECTJ:使用ASPECTJ表達式//FilterType.REGEX:使用正則指定//FilterType.CUSTOM:使用自定義規則public class MainConfig { //給容器中注冊一個Bean;類型為返回值的類型,id默認是用方法名作為id @Bean('person') public Person person01(){ return new Person('lisi', 20); } }
2、@ComponentScan-自動掃描組件&指定掃描規則
@CompoentScan的使用,參考1中以上代碼即可。
3、自定義TypeFilter指定過濾規則
參考如下代碼:
package com.atguigu.config; import java.io.IOException; import org.springframework.core.io.Resource;import org.springframework.core.type.AnnotationMetadata;import org.springframework.core.type.ClassMetadata;import org.springframework.core.type.classreading.MetadataReader;import org.springframework.core.type.classreading.MetadataReaderFactory;import org.springframework.core.type.filter.TypeFilter; public class MyTypeFilter implements TypeFilter { /** * metadataReader:讀取到的當前正在掃描的類的信息 * metadataReaderFactory:可以獲取到其他任何類信息的 */ @Override public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException { // TODO Auto-generated method stub //獲取當前類注解的信息 AnnotationMetadata annotationMetadata = metadataReader.getAnnotationMetadata(); //獲取當前正在掃描的類的類信息 ClassMetadata classMetadata = metadataReader.getClassMetadata(); //獲取當前類資源(類的路徑) Resource resource = metadataReader.getResource(); String className = classMetadata.getClassName(); System.out.println('--->'+className); if(className.contains('er')){ return true; } return false; } }
4、@Scope-設置組件作用域
參考如下代碼:
5、@Lazy懶加載
6、@Conditional-按照條件注冊bean
package com.atguigu.condition; import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;import org.springframework.beans.factory.support.BeanDefinitionRegistry;import org.springframework.context.annotation.Condition;import org.springframework.context.annotation.ConditionContext;import org.springframework.core.env.Environment;import org.springframework.core.type.AnnotatedTypeMetadata; //判斷是否linux系統public class LinuxCondition implements Condition { /** * ConditionContext:判斷條件能使用的上下文(環境) * AnnotatedTypeMetadata:注釋信息 */ public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) { // TODO是否linux系統 //1、能獲取到ioc使用的beanfactory ConfigurableListableBeanFactory beanFactory = context.getBeanFactory(); //2、獲取類加載器 ClassLoader classLoader = context.getClassLoader(); //3、獲取當前環境信息 Environment environment = context.getEnvironment(); //4、獲取到bean定義的注冊類 BeanDefinitionRegistry registry = context.getRegistry(); String property = environment.getProperty('os.name'); //可以判斷容器中的bean注冊情況,也可以給容器中注冊bean boolean definition = registry.containsBeanDefinition('person'); if(property.contains('linux')){ return true; } return false; } }
7、@Import-給容器中快速導入一個組件
源代碼如下:
@Target(ElementType.TYPE)@Retention(RetentionPolicy.RUNTIME)@Documentedpublic @interface Import { /** * {@link Configuration @Configuration}, {@link ImportSelector}, * {@link ImportBeanDefinitionRegistrar}, or regular component classes to import. */ Class<?>[] value(); }
package com.atguigu.config; import org.springframework.beans.factory.config.ConfigurableBeanFactory;import org.springframework.context.annotation.Bean;import org.springframework.context.annotation.Conditional;import org.springframework.context.annotation.Configuration;import org.springframework.context.annotation.Import;import org.springframework.context.annotation.Lazy;import org.springframework.context.annotation.Scope; import com.atguigu.bean.Blue;import com.atguigu.bean.Color;import com.atguigu.bean.ColorFactoryBean;import com.atguigu.bean.Person;import com.atguigu.bean.Red;import com.atguigu.condition.LinuxCondition;import com.atguigu.condition.MyImportBeanDefinitionRegistrar;import com.atguigu.condition.MyImportSelector;import com.atguigu.condition.WindowsCondition;import com.atguigu.test.IOCTest; //類中組件統一設置。滿足當前條件,這個類中配置的所有bean注冊才能生效;@Conditional({ WindowsCondition.class })@Configuration@Import({ Color.class, Red.class, MyImportSelector.class, MyImportBeanDefinitionRegistrar.class })// @Import導入組件,id默認是組件的全類名public class MainConfig2 { // 默認是單實例的 /** * ConfigurableBeanFactory#SCOPE_PROTOTYPE * * @see ConfigurableBeanFactory#SCOPE_SINGLETON * @see org.springframework.web.context.WebApplicationContext#SCOPE_REQUEST * request * @see org.springframework.web.context.WebApplicationContext#SCOPE_SESSION * sesssion @return * @Scope:調整作用域 prototype:多實例的:ioc容器啟動并不會去調用方法創建對象放在容器中。 每次獲取的時候才會調用方法創建對象; * singleton:單實例的(默認值):ioc容器啟動會調用方法創建對象放到ioc容器中。 * 以后每次獲取就是直接從容器(map.get())中拿, request:同一次請求創建一個實例 * session:同一個session創建一個實例 * * 懶加載: 單實例bean:默認在容器啟動的時候創建對象; * 懶加載:容器啟動不創建對象。第一次使用(獲取)Bean創建對象,并初始化; * */ // @Scope('prototype') @Lazy @Bean('person') public Person person() { System.out.println('給容器中添加Person....'); return new Person('張三', 25); } /** * @Conditional({Condition}) : 按照一定的條件進行判斷,滿足條件給容器中注冊bean * * 如果系統是windows,給容器中注冊('bill') * 如果是linux系統,給容器中注冊('linus') */ @Bean('bill') public Person person01() { return new Person('Bill Gates', 62); } @Conditional(LinuxCondition.class) @Bean('linus') public Person person02() { return new Person('linus', 48); } /** * 給容器中注冊組件; * 1)、包掃描+組件標注注解(@Controller/@Service/@Repository/@Component)[自己寫的類] * 2)、@Bean[導入的第三方包里面的組件] 3)、@Import[快速給容器中導入一個組件] * 1)、@Import(要導入到容器中的組件);容器中就會自動注冊這個組件,id默認是全類名 * 2)、ImportSelector:返回需要導入的組件的全類名數組; * 3)、ImportBeanDefinitionRegistrar:手動注冊bean到容器中 4)、使用Spring提供的 * FactoryBean(工廠Bean); 1)、默認獲取到的是工廠bean調用getObject創建的對象 * 2)、要獲取工廠Bean本身,我們需要給id前面加一個& &colorFactoryBean */ @Bean public ColorFactoryBean colorFactoryBean() { return new ColorFactoryBean(); } }
8、@Import-使用ImportSelect
8.1 配置如下:
8.2對應的類需實現ImportSelector接口
package com.atguigu.condition; import org.springframework.context.annotation.ImportSelector;import org.springframework.core.type.AnnotationMetadata; //自定義邏輯返回需要導入的組件public class MyImportSelector implements ImportSelector { //返回值,就是到導入到容器中的組件全類名 //AnnotationMetadata:當前標注@Import注解的類的所有注解信息 public String[] selectImports(AnnotationMetadata importingClassMetadata) { // TODO Auto-generated method stub //importingClassMetadata //方法不要返回null值 return new String[]{'com.atguigu.bean.Blue','com.atguigu.bean.Yellow'}; } }
9、@Import-使用ImportBeanDefinitionRegistrar
9.1配置如下:
9.2對應的類需實現ImportBeanDefinitionRegistrar接口
package com.atguigu.condition; import org.springframework.beans.factory.support.BeanDefinitionRegistry;import org.springframework.beans.factory.support.RootBeanDefinition;import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;import org.springframework.core.type.AnnotationMetadata; import com.atguigu.bean.RainBow; public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar { /** * AnnotationMetadata:當前類的注解信息 * BeanDefinitionRegistry:BeanDefinition注冊類; * 把所有需要添加到容器中的bean;調用 * BeanDefinitionRegistry.registerBeanDefinition手工注冊進來 */ public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) { boolean definition = registry.containsBeanDefinition('com.atguigu.bean.Red'); boolean definition2 = registry.containsBeanDefinition('com.atguigu.bean.Blue'); if(definition && definition2){ //指定Bean定義信息;(Bean的類型,Bean。。。) RootBeanDefinition beanDefinition = new RootBeanDefinition(RainBow.class); //注冊一個Bean,指定bean名 registry.registerBeanDefinition('rainBow', beanDefinition); } } }
10、使用FactoryBean注冊組件
10.1 配置如下:
10.2對應的類需實現org.springframework.beans.factory.FactoryBean<T>接口
package com.atguigu.bean; import org.springframework.beans.factory.FactoryBean; //創建一個Spring定義的FactoryBeanpublic class ColorFactoryBean implements FactoryBean<Color> { //返回一個Color對象,這個對象會添加到容器中 public Color getObject() throws Exception { // TODO Auto-generated method stub System.out.println('ColorFactoryBean...getObject...'); return new Color(); } public Class<?> getObjectType() { // TODO Auto-generated method stub return Color.class; } //是單例? //true:這個bean是單實例,在容器中保存一份 //false:多實例,每次獲取都會創建一個新的bean; public boolean isSingleton() { // TODO Auto-generated method stub return false; } }
11、給容器中注冊組件-總結
/*** 給容器中注冊組件;* 1)、包掃描+組件標注注解(@Controller/@Service/@Repository/@Component)[自己寫的類]* 2)、@Bean[導入的第三方包里面的組件] 3)、@Import[快速給容器中導入一個組件]* 1)、@Import(要導入到容器中的組件);容器中就會自動注冊這個組件,id默認是全類名* 2)、ImportSelector:返回需要導入的組件的全類名數組;* 3)、ImportBeanDefinitionRegistrar:手動注冊bean到容器中 4)、使用Spring提供的* FactoryBean(工廠Bean); 1)、默認獲取到的是工廠bean調用getObject創建的對象* 2)、要獲取工廠Bean本身,我們需要給id前面加一個& &colorFactoryBean*/
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。
相關文章: