7 SpringBoot框架(中):依赖注入详解、SpringBoot中的日志管理、@Value 注解和@ConfigurationProperties 注解加载配置文件

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档

文章目录


前言


提示:以下是本篇文章正文内容,下面案例可供参考

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() {
   
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值