今天开始,我们就来一一突破这些大厂的面试题,好了,开始今天的正文。
问题:
正如题目所说:Spring IOC容器中只存放单例Bean吗?
先给出结论吧
这里,想来想去,我还是直接了当的说吧:是的,Spring IOC容器中只存放单例Bean。接下来,且听我细细道来为哈只存放单例Bean。
问题分析
既然,我们已经知道Spring IOC容器中只存放单例Bean,但是在面试的时候不能只说这一句话呀,否则,面试官就会把你直接Pass掉。为啥?如果你只说这一句话,面试官可能就会认为你是懵的,而且懵对的概率为50%,如果你懵错了,面试官认为你不会,如果你懵对了,面试官有可能也会认为你不会。所以,除了答对结论之外,还要清晰的说出Spring IOC容器中为啥只存放单例Bean。
好了,我们正式开始分析这个问题。
IOC容器初始化的时候,会将所有的bean初始化在singletonObjects这个ConcurrentHashMap中, bean是单例的。
在获取bean的时候,首先会从singletonObjects去取,通过debug,发现如果scope是单例,则可以获取到bean,如果scope是多例,则获取不到bean,需要从一个叫mergedBeanDefinitions的ConcurrentHashMap中去获取bean的定义,然后再根据bean的scope去决定如何创建bean,如果scope=prototype,则每次都会创建一个新的实例。
这里,我们可以大概得出这样的结论:
IOC在初始化时,只会将scope= singleton(单例)的对象进行实例化,而不会去实例化scope=prototype的对象(多例)。
接下来,我们就来debug一下Spring的源码。
首先,我们创建一个用于测试作用域为多例,获取不同实例的Person类,如下所示。
- public class Person {
- @Value("张三")
- private String name;
- @Value("#{20-2}")
- private Integer age;
- @Value("${person.nickName}")
- private String nickName;
- public Person() {
- }
- public Person(String name, Integer age) {
- this.name = name;
- this.age = age;
- }
- //省略get/set
- }
接下来,创建一个MainConfig类,如下所示。
- @Configuration
- public class MainConfig {
- @Bean("person")
- @Scope("prototype")
- public Person person(){
- System.out.println("给容器中添加Person…");
- return new Person("张三", 25);
- }
- }
可以看到,此时MainConfig测试的是作用域为多例,获取不同实例的场景。而如果要想测试作用域为单例,获取相同实例的场景,则只需要将MainConfig类中的person()方法上的 @Scope("prototype")注解去掉即可,如下所示。
- @Configuration
- public class MainConfig {
- @Bean("person")
- public Person person(){
- System.out.println("给容器中添加Person…");
- return new Person("张三", 25);
- }
- }
接下来,再编写一个main方法用于启动测试程序。
- public static void main(String[] args){
- ApplicationContext applicationContext = new AnnotationConfigApplicationContext(MainConfig.class);
- Person person = applicationContext.getBean(Person.class);
- Person person2 = applicationContext.getBean(Person.class);
- if(person.equals(person2)){
- System.out.println("同一个实例");
- }else{
- System.out.println("不同的实例");
- }
- }
启动程序,开始debug测试单例情况。