From f6c097b5e381fe55c0974b21f7d7b5c71e895ccd Mon Sep 17 00:00:00 2001 From: guide Date: Sun, 10 Apr 2022 23:02:55 +0800 Subject: [PATCH] =?UTF-8?q?[docs=20improve]@Autowired=20=E5=92=8C=20@Resou?= =?UTF-8?q?rce=20=E7=9A=84=E5=8C=BA=E5=88=AB=E6=98=AF=E4=BB=80=E4=B9=88?= =?UTF-8?q?=EF=BC=9F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../spring-knowledge-and-questions-summary.md | 162 ++++++++++++------ 1 file changed, 112 insertions(+), 50 deletions(-) diff --git a/docs/system-design/framework/spring/spring-knowledge-and-questions-summary.md b/docs/system-design/framework/spring/spring-knowledge-and-questions-summary.md index 288dd2fb..e1b38ab9 100644 --- a/docs/system-design/framework/spring/spring-knowledge-and-questions-summary.md +++ b/docs/system-design/framework/spring/spring-knowledge-and-questions-summary.md @@ -137,11 +137,11 @@ Spring AOP 已经集成了 AspectJ ,AspectJ 应该算的上是 Java 生态系 如果我们的切面比较少,那么两者性能差异不大。但是,当切面太多的话,最好选择 AspectJ ,它比 Spring AOP 快很多。 -## Spring bean +## Spring Bean -### 什么是 bean? +### 什么是 Spring Bean? -简单来说,bean 代指的就是那些被 IoC 容器所管理的对象。 +简单来说,Bean 代指的就是那些被 IoC 容器所管理的对象。 我们需要告诉 IoC 容器帮助我们管理哪些对象,这个是通过配置元数据来定义的。配置元数据可以是 XML 文件、注解或者 Java 配置类。 @@ -158,50 +158,18 @@ Spring AOP 已经集成了 AspectJ ,AspectJ 应该算的上是 Java 生态系 `org.springframework.beans`和 `org.springframework.context` 这两个包是 IoC 实现的基础,如果想要研究 IoC 相关的源码的话,可以去看看 -### bean 的作用域有哪些? +### 将一个类声明为 Bean 的注解有哪些? -Spring 中 Bean 的作用域通常有下面几种: - -- **singleton** : 唯一 bean 实例,Spring 中的 bean 默认都是单例的,对单例设计模式的应用。 -- **prototype** : 每次请求都会创建一个新的 bean 实例。 -- **request** : 每一次 HTTP 请求都会产生一个新的 bean,该 bean 仅在当前 HTTP request 内有效。 -- **session** : 每一次来自新 session 的 HTTP 请求都会产生一个新的 bean,该 bean 仅在当前 HTTP session 内有效。 -- **global-session** : 全局 session 作用域,仅仅在基于 portlet 的 web 应用中才有意义,Spring5 已经没有了。Portlet 是能够生成语义代码(例如:HTML)片段的小型 Java Web 插件。它们基于 portlet 容器,可以像 servlet 一样处理 HTTP 请求。但是,与 servlet 不同,每个 portlet 都有不同的会话。 - -**如何配置 bean 的作用域呢?** - -xml 方式: - -```xml - -``` - -注解方式: - -```java -@Bean -@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE) -public Person personPrototype() { - return new Person(); -} -``` - -### 单例 bean 的线程安全问题了解吗? - -大部分时候我们并没有在项目中使用多线程,所以很少有人会关注这个问题。单例 bean 存在线程问题,主要是因为当多个线程操作同一个对象的时候是存在资源竞争的。 - -常见的有两种解决办法: - -1. 在 bean 中尽量避免定义可变的成员变量。 -2. 在类中定义一个 `ThreadLocal` 成员变量,将需要的可变成员变量保存在 `ThreadLocal` 中(推荐的一种方式)。 - -不过,大部分 bean 实际都是无状态(没有实例变量)的(比如 Dao、Service),这种情况下, bean 是线程安全的。 +- `@Component` :通用的注解,可标注任意类为 `Spring` 组件。如果一个 Bean 不知道属于哪个层,可以使用`@Component` 注解标注。 +- `@Repository` : 对应持久层即 Dao 层,主要用于数据库相关操作。 +- `@Service` : 对应服务层,主要涉及一些复杂的逻辑,需要用到 Dao 层。 +- `@Controller` : 对应 Spring MVC 控制层,主要用户接受用户请求并调用 Service 层返回数据给前端页面。 ### @Component 和 @Bean 的区别是什么? -1. `@Component` 注解作用于类,而`@Bean`注解作用于方法。 -2. `@Component`通常是通过类路径扫描来自动侦测以及自动装配到 Spring 容器中(我们可以使用 `@ComponentScan` 注解定义要扫描的路径从中找出标识了需要装配的类自动装配到 Spring 的 bean 容器中)。`@Bean` 注解通常是我们在标有该注解的方法中定义产生这个 bean,`@Bean`告诉了 Spring 这是某个类的实例,当我需要用它的时候还给我。 -3. `@Bean` 注解比 `@Component` 注解的自定义性更强,而且很多地方我们只能通过 `@Bean` 注解来注册 bean。比如当我们引用第三方库中的类需要装配到 `Spring`容器时,则只能通过 `@Bean`来实现。 +- `@Component` 注解作用于类,而`@Bean`注解作用于方法。 +- `@Component`通常是通过类路径扫描来自动侦测以及自动装配到 Spring 容器中(我们可以使用 `@ComponentScan` 注解定义要扫描的路径从中找出标识了需要装配的类自动装配到 Spring 的 bean 容器中)。`@Bean` 注解通常是我们在标有该注解的方法中定义产生这个 bean,`@Bean`告诉了 Spring 这是某个类的实例,当我需要用它的时候还给我。 +- `@Bean` 注解比 `@Component` 注解的自定义性更强,而且很多地方我们只能通过 `@Bean` 注解来注册 bean。比如当我们引用第三方库中的类需要装配到 `Spring`容器时,则只能通过 `@Bean`来实现。 `@Bean`注解使用示例: @@ -240,16 +208,110 @@ public OneService getService(status) { } ``` -### 将一个类声明为 bean 的注解有哪些? +### @Autowired 和 @Resource 的区别是什么? -我们一般使用 `@Autowired` 注解自动装配 bean,要想把类标识成可用于 `@Autowired` 注解自动装配的 bean 的类,采用以下注解可实现: +`Autowired` 属于 Spring 内置的注解,默认的注入方式为`byType`(根据类型进行匹配),也就是说会优先根据接口类型去匹配并注入 Bean (接口的实现类)。 -- `@Component` :通用的注解,可标注任意类为 `Spring` 组件。如果一个 Bean 不知道属于哪个层,可以使用`@Component` 注解标注。 -- `@Repository` : 对应持久层即 Dao 层,主要用于数据库相关操作。 -- `@Service` : 对应服务层,主要涉及一些复杂的逻辑,需要用到 Dao 层。 -- `@Controller` : 对应 Spring MVC 控制层,主要用户接受用户请求并调用 Service 层返回数据给前端页面。 +**这会有什么问题呢?** 当一个接口存在多个实现类的话,`byType`这种方式就无法正确注入对象了,因为这个时候 Spring 会同时找到多个满足条件的选择,默认情况下它自己不知道选择哪一个。 -### bean 的生命周期? +这种情况下,注入方式会变为 `byName`(根据名称进行匹配),这个名称通常就是类名(首字母小写)。就比如说下面代码中的 `smsService` 就是我这里所说的名称,这样应该比较好理解了吧。 + +```java +// smsService 就是我们上面所说的名称 +@Autowired +private SmsService smsService; +``` + +举个例子,`SmsService` 接口有两个实现类: `SmsServiceImpl1`和 `SmsServiceImpl2`,且它们都已经被 Spring 容器所管理。 + +```java +// 报错,byName 和 byType 都无法匹配到 bean +@Autowired +private SmsService smsService; +// 正确注入 SmsServiceImpl1 对象对应的 bean +@Autowired +private SmsService smsServiceImpl1; +// 正确注入 SmsServiceImpl1 对象对应的 bean +// smsServiceImpl1 就是我们上面所说的名称 +@Autowired +@Qualifier(value = "smsServiceImpl1") +private SmsService smsService; +``` + +我们还是建议通过 `@Qualifier` 注解来显示指定名称而不是依赖变量的名称。 + +`@Resource`属于 JDK 提供的注解,默认注入方式为 `byName`。如果无法通过名称匹配到对应的实现类的话,注入方式会变为`byType`。 + +`@Resource` 有两个比较重要且日常开发常用的属性:`name`(名称)、`type`(类型)。 + +```java +public @interface Resource { + String name() default ""; + Class type() default Object.class; +} +``` + +如果仅指定 `name` 属性则注入方式为`byName`,如果仅指定`type`属性则注入方式为`byType`,如果同时指定`name` 和`type`属性(不建议这么做)则注入方式为`byType`+`byName`。 + +```java +// 报错,byName 和 byType 都无法匹配到 bean +@Resource +private SmsService smsService; +// 正确注入 SmsServiceImpl1 对象对应的 bean +@Autowired +private SmsService smsServiceImpl1; +// 正确注入 SmsServiceImpl1 对象对应的 bean(比较推荐这种方式) +@Autowired +@Resource(name = "smsServiceImpl1") +private SmsService smsService; +``` + +简单总结一下: + +- `@Autowired` 是 Spring 提供的注解,`@Resource` 是 JDK 提供的注解。 +- `Autowired` 默认的注入方式为`byType`(根据类型进行匹配),`@Resource`默认注入方式为 `byName`(根据名称进行匹配)。 +- 当一个类存在多个实现类的情况下,`@Autowired` 和`@Resource`都需要通过名称进行匹配才能正确匹配到对应的 Bean。`Autowired` 可以通过 `@Qualifier` 注解来显示指定名称,`@Resource`可以通过 `name` 属性来显示指定名称。 + +### Bean 的作用域有哪些? + +Spring 中 Bean 的作用域通常有下面几种: + +- **singleton** : 唯一 bean 实例,Spring 中的 bean 默认都是单例的,对单例设计模式的应用。 +- **prototype** : 每次请求都会创建一个新的 bean 实例。 +- **request** : 每一次 HTTP 请求都会产生一个新的 bean,该 bean 仅在当前 HTTP request 内有效。 +- **session** : 每一次来自新 session 的 HTTP 请求都会产生一个新的 bean,该 bean 仅在当前 HTTP session 内有效。 +- **global-session** : 全局 session 作用域,仅仅在基于 portlet 的 web 应用中才有意义,Spring5 已经没有了。Portlet 是能够生成语义代码(例如:HTML)片段的小型 Java Web 插件。它们基于 portlet 容器,可以像 servlet 一样处理 HTTP 请求。但是,与 servlet 不同,每个 portlet 都有不同的会话。 + +**如何配置 bean 的作用域呢?** + +xml 方式: + +```xml + +``` + +注解方式: + +```java +@Bean +@Scope(value = ConfigurableBeanFactory.SCOPE_PROTOTYPE) +public Person personPrototype() { + return new Person(); +} +``` + +### 单例 Bean 的线程安全问题了解吗? + +大部分时候我们并没有在项目中使用多线程,所以很少有人会关注这个问题。单例 Bean 存在线程问题,主要是因为当多个线程操作同一个对象的时候是存在资源竞争的。 + +常见的有两种解决办法: + +1. 在 Bean 中尽量避免定义可变的成员变量。 +2. 在类中定义一个 `ThreadLocal` 成员变量,将需要的可变成员变量保存在 `ThreadLocal` 中(推荐的一种方式)。 + +不过,大部分 Bean 实际都是无状态(没有实例变量)的(比如 Dao、Service),这种情况下, Bean 是线程安全的。 + +### Bean 的生命周期了解么? > 下面的内容整理自: ,除了这篇文章,再推荐一篇很不错的文章 :