谈谈你对Spring Bean的理解

大家好,我是被编程耽误的文艺Tom。

前几天,有位同学向我反馈,说在面试中问到这样这样一个面试题:谈谈你对Spring Bean的理解。今天咱们就针对这样一个面试题,给大家做一个详细的介绍。我一共分三段来介绍,首先,介绍什么是Spring Bean?然后,定义Spring Bean有哪些方式?,最后,给大家介绍Spring容器是如何加载Bean的?

咱们先来看什么是Spring Bean?

1、什么是Spring Bean?
Spring Bean是Spring中最基本的组成单元,Spring 官方文档对 Bean 的解释是这样的:
In Spring, the objects that form the backbone of your application and that are managed by the Spring IoC container are called beans. A bean is an object that is instantiated, assembled, and managed by a Spring IoC container.
翻译过来就是:
在 Spring 中,构成应用程序主干并由Spring IoC容器管理的对象称为Bean。Bean是一个由Spring IoC容器实例化、组装和管理的对象。

从官方定义中,我们可以提取出以下信息:
1、Bean是对象,一个或者多个不限定。
2、Bean托管在Spring中一个叫IoC的容器中。
3、我们的程序是由一个个Bean构成的。
Spring是通过声明式配置的方式来定义Bean的,所有创建Bean需要的前置依赖或者参数都是通过配置文件先声明,Spring启动以后会解析这些声明好的配置内容。那么,我们该如何去定义Spring中的Bean呢?

2、定义Spring Bean有哪些方式?
一般来说,Spring Bean的定义配置有三种方式:
第一种:基于XML的方式配置
这种配置方式主要适用于以下两类场景:
1、Bean实现类来自第三方类库、比如DataSource等
2、需要定义命名空间的配置,如:context、aop、mvc等。
举个例子,来看一段代码

//导入其他配置文件Bean的定义

 


  


这段代码是标准的Spring配置内容,咱们从上往下来看,首先导入一个叫做resource1.xml的标准配置文件,然后,导入了第二个标准的配置文件resource2.xml,接着,定义了一个叫做userService的Bean,对应的类是com.gupaoedu.vip.UserService,并且声明了在userService实例化之后要调用init()方法,在userService销毁之后调用destory()方法。
第二种:基于注解扫描的方式配置
这种配置方式主要适用于:在开发中需要引用的类,如Controller、Service、Dao等。
两种配置方法:

  //包含的目标类
     //排除的目标类

在这段配置中,context:component-scan相当于使用了Spring内置的扫描注解的组件 @ComponentScan ,声明了需要扫描的基础包路径com.gupaoedu.vip,把所有com.gupaoedu.vip.mall下面的子包全部纳入扫描范围。并且排除了com.gupaoedu.vip.mall 包下面所有以Controller结尾或者包含Controller结尾的类。

使用Spring容器管理组件的beanName规则,默认是类名首字母变小写,可以自己修改。Spring主要提供了4种内置的注解用来声明Bean。它们分别是@Controller,声明为控制层组件的Bean,@Service,声明为业务逻辑层组件的Bean,@Repository,声明为数据访问层组件的Bean,当对组件的层次难以定位的时候使用@Component注解来声明。

Spring提供了四个注解,这些注解的作用与上面的XML定义bean效果一致,在于将组件交给Spring容器管理。组件的名称默认是类名(首字母变小写),可以自己修改:

第三种:基于Java类的配置
这种配置方式主要适用于以下两类场景:
1、需要通过代码控制对象创建逻辑的场景;
2、实现零配置,消除XML配置文件的场景。
同样还是举个例子,来看一段代码:
@Configuration
public class BeansConfiguration {


@Bean  
public Student student(){
    
    Student student = new Student();
    student.setName("张三");
    student.setTeacher(teacher());
    return student;
    
}  

@Bean  
public Teacher teacher(){
    
    Teacher teacher = new Teacher();
    teacher.setName("Tom");
    return teacher;
    
}

}
通过分析这段代码,我可以了解到使用基于类的配置需要以下步骤:
1、首先,在BeansConfiguration类上配置@Configuration注解,表示将BeansConfiguration类将定义Bean的元数据
2、在方法上使用@Bean注解,方法名默认就是Bean的名称,方法返回值就是Bean的实例。
3、通过AnnotationConfigApplicationContext或子类来启动Spring容器,从而加载这些已经声明好的注解配置。

最后,总结一下,定义Spring Bean三种方式,分别是:基于XML的方式配置、基于注解扫描的方式配置、基于元数据类的配置。
那么Bean的配置定义好以后,Spring又是如何加载这些Bean配置并创建Bean实例呢?
3、Spring容器如何加载Bean?
Spring解析这些声明好的配置内容,将这些配置内容都转化为BeanDefinition对象,BeanDefinition中几乎保存了配置文件中声明的所有内容,再将BeanDefinition存到一个叫做beanDefinitionMap中。以beanName作为Key,以BeanDefinition对象作为Value。之后Spring容器,根据beanName找到对应的BeanDefinition,再去选择具体的创建策略。而Spring具体的创建策略如图所示,大家想要高清无码图的可以在评论区留言。

关于Spring Bean的面试题解析就到这里,相信大家也已经理解。如果想看Spring BeanDefiniton定义的小伙伴,可以关注我后续的更新内容。

我是被编程耽误的文艺Tom,如果大家还有其他疑问,也可以在评论区留言。如果我的解析对你有帮助,请动动手指一键三连分享给更多的人。

关注我,面试不再难!

关注微信公众号『 Tom弹架构 』回复“简历”可获取配套资料。

  • 本文为“Tom弹架构”原创,转载请注明出处。技术在于分享,我分享我快乐!如果您有任何建议也可留言评论或私信,您的支持是我坚持创作的动力。关注微信公众号『 Tom弹架构 』可获取更多技术干货!

**原创不易,坚持很酷,都看到这里了,小伙伴记得点赞、收藏、在看,一键三连加关注!如果你觉得内容太干,可以分享转发给朋友滋润滋润!
**

你可能感兴趣的:(spring)