提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档
文章目录
前言
提示:以下是本篇文章正文内容,下面案例可供参考
1 SpringBoot中依赖注入(详解)
1.1 问题1:使用依赖注入注入对象时,注入的是无参的构造方法(不建议使用构造方法那种依赖注入方式),那么无法通过构造方法传递参数进来了,怎么处理?使用@Value注解
- 在python开发中经常要在构造方法处传参数进来做一些初始化操作,例如初始化一些配置文件。
但是在java的SpringBoot开发中依赖注入中一条规则就是,需要被注入的类只能有无参构造方法;因为采用构造方法注入有一个巨大缺陷,全部是构造方法注入会导致循环依赖不能解决,最后报错。所以我们现在更加喜欢使用字段注入,这就丧失了一个很重要的功能。
其实,SpringBoot也想到了这个问题,使用@Value注解就可以处理这个问题
在 Spring Boot 中,@Value 注解是一个非常实用的注解,主要用于将外部配置的值注入到 Bean 的字段、方法参数或者构造函数参数中。下面从基本用法、支持的表达式、注入不同类型的值以及与其他注解的配合使用等方面详细介绍 @Value 注解的使用方法。
@@Value 注解会放在下面一节中详细讲述
1.2 问题2:使用依赖注入一个接口时,如果接口有多个实现类,是注入哪一个实现类呢?
1.2.1 单实现类的情况
Spring Boot 依赖注入一个接口,如果接口只有单个实现类,Spring Boot 会自动注入该实现类,您不需要做任何额外的配置。Spring 容器会根据类型自动找到唯一的实现类并注入。
示例:只有一个实现类,直接使用 @Autowired
注解依赖注入即可
假设您只有一个实现类:
1.2.1.1 字段注入
public interface MyService {
void execute();
}
@Service
public class MyServiceImpl implements MyService {
@Override
public void execute() {
System.out.println("Executing MyServiceImpl");
}
}
@RestController
public class MyController {
@Autowired
private MyService myService; // 字段注入
@GetMapping("/test")
public void test() {
myService.execute(); // "Executing MyServiceImpl"
}
}
1.2.1.2 构造方法注入
构造方法注入有一个巨大缺陷,全部是构造方法注入会导致循环依赖不能解决,最后报错。所以我们现在更加喜欢使用字段注入
public interface MyService {
void execute();
}
@Service
public class MyServiceImpl implements MyService {
@Override
public void execute() {
System.out.println("Executing MyServiceImpl");
}
}
@RestController
public class MyController {
private final MyService myService;
@Autowired
public MyController(MyService myService) {
this.myService = myService;
}
@GetMapping("/test")
public void test() {
myService.execute(); // "Executing MyServiceImpl"
}
}
1.2.2 多实现类的情况
1.2.2.1 使用 @Qualifier 指定注入的实现类
当你有多个实现类时,@Qualifier
可以帮助 Spring 确定注入哪个特定的实现类。
@Qualifier注解不能单独使用,必须配合@Autowired使用
1.2.2.1.1 字段注入
public interface MyService {
void execute();
}
@Service
public class MyServiceImplA implements MyService {
@Override
public void execute() {
System.out.println("Executing MyServiceImplA");
}
}
@Service
public class MyServiceImplB implements MyService {
@Override
public void execute() {
System.out.println("Executing MyServiceImplB");
}
}
@RestController
public class MyController {
@Autowired
@Qualifier("myServiceImplB")
// 使用 @Qualifier 来指定注入 MyServiceImplB
// 这里为什么类名首字母小写了,具体原因看SpringBoot注入中默认的Bean名称规则
private MyService myService;
@GetMapping("/test")
public void test() {
myService.execute(); // "Executing MyServiceImplB"
}
}
说明:
@Autowired
:这依然是用来让 Spring 自动注入依赖。@Qualifier("myServiceImplB")
:这个注解指定了要注入MyServiceImplB
,根据在@Service
注解中定义的类名。- 这样,Spring 会将
MyServiceImplB
自动注入到myService
字段。
关键点:
- 使用字段注入时,
@Qualifier
依然与@Autowired
一起使用。 @Qualifier
的参数值需要与@Service
注解中给定的 Bean 名称相匹配,或者是默认类名的小写形式(例如myServiceImplB
)。- 这样就完成了字段注入并且明确指定了使用的实现类!
1.2.2.1.2 构造方法注入
public interface MyService {
void execute();
}
@Service
public class MyServiceImplA implements MyService {
@Override
public void execute() {
System.out.println("Executing MyServiceImplA");
}
}
@Service
public class MyServiceImplB implements MyService {
@Override
public void execute() {
System.out.println("Executing MyServiceImplB");
}
}
@RestController
public class MyController {
private final MyService myService;
@Autowired
public MyController(@Qualifier("myServiceImplB") MyService myService) {
this.myService = myService;
}
@GetMapping("/test")
public void test() {
myService.execute(); // "Executing MyServiceImplB"
}
}
在上述例子中,@Qualifier("myServiceImplB")
用来指定注入 MyServiceImplB
实现类。如果没有使用 @Qualifier
,Spring 就无法知道注入哪个实现类。
1.2.2.1.3 SpringBoot 注解中默认 Bean 名称规则
@Service
注解中的默认 Bean 名称确实是基于类名的小写形式,而不是直接使用类名。我们可以更清楚地解释这一点:
默认 Bean 名称规则
- 当使用
@Service
注解时,如果你没有显式指定@Service
注解的value
属性(即 Bean 名称),Spring 会自动将 Bean 的名称设置为该类名的首字母小写形式。 - 比如:
MyServiceImplA
的默认 Bean 名称是myServiceImplA
(首字母小写)。MyServiceImplB
的默认 Bean 名称是myServiceImplB
。
@Component
,@Controller
等等这些注解都有这个特性。
直接使用类名的情况
如果你在 @Service
中没有指定 value
,Spring 会使用自动生成的默认 Bean 名称(即类名首字母小写)作为 Bean 的标识符,而不是直接使用类名。
示例:
假设我们有以下代码:
@Service
public class MyServiceImplA implements MyService {
@Override
public void execute() {