Section 03 - XML与注解批量注册Bean
@CompponmentScan 代替@Bean 分别新增controller,service,dao三个package,并增加响应的类,通过@Controller,@Service,@Repository三个注解将三个类声明为Bean
@Controller
public class PersonController {
}
@Service
public class PersonService {
}
@Repository
public class PersonDao {
}
修改BeanConfig,增加@ComponentScan注解,将com.citi包下面的所有Bean都扫描到BeanConfig这个配置类中,就相当于XML配置文件中有许多bean标签,好处是不用一个个写bean标签,通过一个注解可以扫描所有的Bean
@Configuration
@ComponentScan(basePackages = “com.citi”)
public class BeanConfig {
}
测试IoC容器是否实例化扫描到的Bean,新增一个ComponentScanTest测试类
public class ComponentScanTest {
@Test
public void getBeansByScan(){
ApplicationContext context = new AnnotationConfigApplicationContext(BeanConfig.class);
String[] beanDefinitionNames = context.getBeanDefinitionNames();
for (String beanDefinitionName : beanDefinitionNames) {
System.out.println(beanDefinitionName);
}
}
}
控制台输出如下,自定义的三个类,包括BeanConfi本身都被容器实例化
entity包中的Person类没有被实例化,是因为Person类上没有添加注解,也即是没有配置bean标签,所以没有注册到容器中,也就没有被实例化,给Person实体类添加一个@Component注解,标识为一个Bean,再次执行测试方法,控制台打印结果如下,Person类被实例化
@ComponentScan源码
includeFilters()的使用
includeFilters()和excludeFilters()返回都是一个Filter数组
FilterType是一个枚举类,默认是ANNOTATION,注解方式
修改BeanConfig代码,增加includeFilters(),只扫描com.citi包下面的@Controller,@Service注解标识的Bean,一定不要忘记useDefaultFilters = false
@Configuration
@ComponentScan(basePackages = “com.citi”,
includeFilters = {
@ComponentScan.Filter(type = FilterType.ANNOTATION, classes = {Controller.class, Service.class})
},
useDefaultFilters = false)
public class BeanConfig {
}
执行测试,控制台输出只包含了@Controller,@Service注解的Bean,@Componet和@Repository注解标识的类并没有被容器管理
Plus:为什么使用includeFilters时,要配置useDefaultFilters = false?
当useDefaultFilters = true时,进入到registerDefaultFilters()方法中
该方法中将@Component注解的Bean都加入到includeFilters中,而@Controller,@Service,@Repository都是基于@Component的注解,所有useDefaultFilters = true会导致设置是扫描@Controller和@Service注解的Bean不生效
使用excludeFilters(),将BeanConfig中includeFilters改为excludeFilters(),并且useDefaultFilters = true,执行测试,可以看出只包含了除了@Controller,@Service标注的Bean之外的其他Bean的实例化对象
使用FilterType枚举类中的ASSIGNABLE_TYPE进行过滤Bean,修改BeanConfig为如下,表示触PersonController和PersonService之外的其他Bean
@Configuration
@ComponentScan(basePackages = “com.citi”,
excludeFilters = {
@ComponentScan.Filter(type = FilterType.ASSIGNABLE_TYPE, classes = {PersonController.class, PersonService.class})
},
useDefaultFilters = true)
public class BeanConfig {
}
使用自定义的FilterType,在config包下新增CustTypeFilter,继承TypeFilter
public class CustTypeFilter implements TypeFilter {
/**
-
@param metadataReader 读取到当前正在扫描的类的信息
-
@param metadataReaderFactory 可以获取到其他任何类的信息
-
@return
-
@throws IOException
*/
@Override
public boolean match(MetadataReader metadataReader, MetadataReaderFactory metadataReaderFactory) throws IOException {
// 获取当前类的注解信息
AnnotationMetadata annotationMetadata = metadataReader.getAnnotationMetadata();
// 获取当前正在扫描的类信息
ClassMetadata classMetadata = metadataReader.getClassMetadata();
// 获取当前类资源(类的路径)
Resource resource = metadataReader.getResource();
// 获取类的名称
String className = classMetadata.getClassName();
// 扫描的类
System.out.println(“------>” + className);
// 过滤条件,return true表示过滤掉符合这个条件的Bean
if (className.contains(“service”)){
return true;
}
return false;
}
}
除了@ComponentScan之外还可以添加@ComponentScans,也就是多个@ComponentScan
使用了两个@ComponentScan注解,一个是去除自定义的Bean,即bean id包含“service”字符串的Bean,另外一个使用Annotation 类型的Filter,去除@Controller注解标记的Bean
@ComponentScans({@ComponentScan(basePackages = “com.citi”,
excludeFilters = {
@ComponentScan.Filter(type = FilterType.CUSTOM, classes = {CustTypeFilter.class})
},
useDefaultFilters = true),
@ComponentScan(basePackages = “com.citi”,
excludeFilters = {
@ComponentScan.Filter(type = FilterType.ANNOTATION, classes = {Controller.class})
},
useDefaultFilters = true)})
public class BeanConfig {
}
这里控制台应该输出包含除了PeresonController和PersonService外的实例化对象,根据控制台打印可以看出配置的两个@ComponentScan都没有生效,这里还有待继续探索,
Section 04 - Bean的单实例和多实例
IoC容器不管是使用注解还是xml方式,默认都是单实例的,IoC容器初始化的时候就会被创建,多实例仅当bean被使用的时候才会创建 修改BeanConfi为
@Configuration
public class BeanConfig {
@Bean
public Person person(){
Person person = new Person();
person.setName(“Stark”);
person.setAge(40);
return person;
}
}
新增测试类SingleOrMultInstanceTest,执行testSingleBean()方法
public class SingleOrMultiInstanceTest {
@Test
public void testSingleBean(){
ApplicationContext context = new AnnotationConfigApplicationContext(BeanConfig.class);
Person person = (Person) context.getBean(“person”);
Person person1 = (Person) context.getBean(“person”);
System.out.println(“是否为单实例:” + (person == person1));
}
}
查看输出结果,通过比较从容器中获取的两个bean是否为一个bean,默认是单实例
@Scope注解可以配置Bean为单实例或者多实例 @Scope源码
修改BeanConfig
@Configuration
public class BeanConfig {
@Bean
@Scope(“prototype”)
public Person person(){
Person person = new Person();
person.setName(“Stark”);
person.setAge(40);
return person;
}
}
执行测试,此时已经为多实例
Section 05 - 懒加载@Lazy
修改BeanConfig,增加@Lazy注解,懒加载指针多单实例Bean,因为单实例是在容器初始化是就创建对象,增加@Lazy注解后,当调用getBean()获取实例化对象时容器才会实例化Bean 首先将BeanConfig中@Scope注解注释
@Configuration
public class BeanConfig {
//@Lazy
@Bean
//@Scope(“prototype”)
public Person person(){
System.out.println(“Bean被实例化”);
最后
这里我特地整理了一份《Android开发核心知识点笔记》,里面就包含了自定义View相关的内容
除了这份笔记,还给大家分享 Android学习PDF+架构视频+面试文档+源码笔记,高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这几块的内容。非常适合近期有面试和想在技术道路上继续精进的朋友。
分享上面这些资源,希望可以帮助到大家提升进阶,如果你觉得还算有用的话,不妨把它们推荐给你的朋友~
喜欢本文的话,给我点个小赞、评论区留言或者转发支持一下呗~
首先将BeanConfig中@Scope注解注释
@Configuration
public class BeanConfig {
//@Lazy
@Bean
//@Scope(“prototype”)
public Person person(){
System.out.println(“Bean被实例化”);
最后
这里我特地整理了一份《Android开发核心知识点笔记》,里面就包含了自定义View相关的内容
[外链图片转存中…(img-I9yrShTm-1720099737423)]
除了这份笔记,还给大家分享 Android学习PDF+架构视频+面试文档+源码笔记,高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料这几块的内容。非常适合近期有面试和想在技术道路上继续精进的朋友。
[外链图片转存中…(img-Nnljnk5F-1720099737424)]
分享上面这些资源,希望可以帮助到大家提升进阶,如果你觉得还算有用的话,不妨把它们推荐给你的朋友~
喜欢本文的话,给我点个小赞、评论区留言或者转发支持一下呗~