• IoC & DI


    Spring 的两大核心思想 : IoC 和 AOP

    我们要将对象的控制权交给Spring ,我们就需要告诉 Spring 哪些对象是需要帮我们进行创建的,这里有两类注解可以实现 :

    类注解(@Controller @Service @Repository @Component @Configuration)和方法注解(@Bean)

    这五大注解都表示把这个对象交给 Spring 进行管理和创建

    五大注解和实际项目开发的关系 : 

    @Controller @Service @Repository 和三层架构有关系

    Controller : 接收请求并返回响应

    Service : 接收完请求具体要做些什么样的事情,真正业务的逻辑处理是什么都是 Service 做的

    Dao : 对数据进行处理

    @Component 存储组件相关,@Configuration 存储配置相关

    我们查看五个类注解的原码 : @Controller @Service @Repository @Component @Configuration

    这四个注解都是基于 @Component 实现的,可以认为是 @Component 的衍生类

    但是是不是说我们所有程序都使用 @Component 就行呢,那是当然不行的,那搞这么多注解就没意义了

    我们用代码尝试一下能否使用 @Service 代替 @Controller 呢?@Controller 是接收请求返回响应,看看换成 @Service 是否还能接收请求返回响应

    1. package com.example.ioc.controller;
    2. import org.springframework.stereotype.Controller;
    3. import org.springframework.web.bind.annotation.RequestBody;
    4. import org.springframework.web.bind.annotation.RequestMapping;
    5. import org.springframework.web.bind.annotation.ResponseBody;
    6. @Controller
    7. @ResponseBody
    8. public class TestController {
    9. @RequestMapping("/test")
    10. public String test(){
    11. return "测试Controller 和其他注解的区别";
    12. }
    13. }

    现在我们把 @Controller 换成 @Service ,我们发现报错了

    这是因为 Spring 对 Controller 进行了一些处理,我们想接收请求就只能使用 Controller ,就是说Controller必须作为我们程序的第一关,这是规范,不能使用其他注解

    那么 @Service 这些可以使用 @Component 来代替呢? 这是可以的,但是并不建议

    但是明明 @Service 和 @Controller 的原码明明差不多,为什么应用上有区别呢?这是因为原码只是注解的声明,只是声明了一个注解,但是 Spring 对于这个注解还会赋予一些别的功能,这些功能在 Spring 的原码里面

    五大注解只能加在类上,并且只能加在自己的代码上,如果我引入了一个 jar包,也希望交给Spring 管理,这是没有办法加五大注解的,这时候就要使用 @Bean

    @Bean 是方法注解 , @Bean 需要搭配五大注解来使用

    使用 @Bean 注解时,bean 的名称是方法名

    五大注解是,bean 的名称是类名的首字母转为小写,如果前两个字母都为大写,bean 的名称就是类名

    还有另一个场景可以使用 @Bean ,对于一个类,我们需要定义多个对象时,比如数据库操作,定义多个数据源

    因为当我们用别的注解的时候,拿到的是同一个数据,地址都一样,我们拿 @Configuration 举例

    1. package com.example.ioc.config;
    2. import org.springframework.context.annotation.Configuration;
    3. @Configuration
    4. public class UserConfig {
    5. public void doConfig(){
    6. System.out.println("do configurarion...");
    7. }
    8. }
    1. UserConfig userConfig = context.getBean(UserConfig.class);
    2. userConfig.doConfig();
    3. System.out.println("userConfig:"+userConfig);
    4. UserConfig userConfig1 = context.getBean(UserConfig.class);
    5. System.out.println("userConfig:"+userConfig1);
    6. System.out.println("userConfig == userConfig1"+(userConfig==userConfig1));

     我们发现,两个数据的地址是完全相同的,它们是否相等的判定返回也是 true 

    接下来我们用 @Bean 试试看

    1. package com.example.ioc.config;
    2. import lombok.Data;
    3. @Data
    4. //假如这是第三方包下的一个类
    5. //现在我在我的项目中用到了 UserInfo,并且我需要用它定义多个对象
    6. public class UserInfo {
    7. private Integer id;
    8. private String name;
    9. private Integer age;
    10. }
    1. package com.example.ioc.config;
    2. import org.springframework.context.annotation.Bean;
    3. import org.springframework.context.annotation.Configuration;
    4. @Configuration
    5. public class BeanConfig {
    6. @Bean
    7. public UserInfo userInfo(){
    8. UserInfo userInfo = new UserInfo();
    9. userInfo.setId(1);
    10. userInfo.setName("zhangsan");
    11. userInfo.setAge(12);
    12. return userInfo;
    13. }
    14. @Bean
    15. public UserInfo userInfo2(){
    16. UserInfo userInfo = new UserInfo();
    17. userInfo.setId(2);
    18. userInfo.setName("wangwu");
    19. userInfo.setAge(13);
    20. return userInfo;
    21. }
    22. }
    1. package com.example.ioc;
    2. import com.example.ioc.component.UserComponent;
    3. import com.example.ioc.config.UserConfig;
    4. import com.example.ioc.config.UserInfo;
    5. import com.example.ioc.controller.UserController;
    6. import com.example.ioc.repo.UserRepository;
    7. import com.example.ioc.service.UserService;
    8. import org.springframework.boot.SpringApplication;
    9. import org.springframework.boot.autoconfigure.SpringBootApplication;
    10. import org.springframework.context.ApplicationContext;
    11. import org.springframework.context.annotation.Bean;
    12. @SpringBootApplication
    13. public class DemoApplication {
    14. public static void main(String[] args) {
    15. //Spring 上下文
    16. ApplicationContext context = SpringApplication.run(DemoApplication.class, args);
    17. //返回的就是 Spring 的运行环境
    18. //@Bean演示
    19. UserInfo userInfo = (UserInfo) context.getBean("userInfo");
    20. System.out.println(userInfo);
    21. UserInfo userInfo2 = (UserInfo) context.getBean("userInfo2");
    22. System.out.println(userInfo2);
    23. }
    24. }

    这样就能返回两个不一样的内容了 

    当一个类型存在多个 bean 时,我们就不能使用类型来获取对象了,就要使用名称进行获取了,或者是名称+类型进行获取

    1. UserInfo userInfo1 = context.getBean("userInfo",UserInfo.class);
    2. System.out.println(userInfo1);

    这样依然可以拿到 

    当我们想进行传参,代码改为 :

    1. package com.example.ioc.config;
    2. import org.springframework.context.annotation.Bean;
    3. import org.springframework.context.annotation.Configuration;
    4. @Configuration
    5. public class BeanConfig {
    6. @Bean
    7. public String name(){
    8. return "wangwu";
    9. }
    10. @Bean
    11. public UserInfo userInfo(String name){
    12. UserInfo userInfo = new UserInfo();
    13. userInfo.setId(1);
    14. userInfo.setName(name);
    15. userInfo.setAge(12);
    16. return userInfo;
    17. }
    18. }

    运行结果如下 

    上面代码我把 public String name 改为 public String name2 依然可以被拿到,但是如果name和name2都存在,那么就根据名称去匹配,需要哪个就把名称放进传参的括号里

    DI 依赖注入也可以叫做"属性装配"或者"依赖装配"

    依赖可以简单认为是属性,比如下面代码中,userService 可以看做是 UserController 的属性

    我们该如何完成依赖注入呢?

    1.属性注入

    我们用 @Autowired,这样就能把 Service 给引进来

    属性注入以类型进行匹配,与注入的属性名称无关,但是如果一个类型存在多个对象时,优先进行名称匹配,如果名称都匹配不上,那就报错

    @Autowired 无法注入一个被 final 修饰的属性(很少遇到这种情况)

    1. @Controller
    2. //用 @Controller 告诉 Spring 帮我们管理这个对象
    3. public class UserController {
    4. @Autowired
    5. private UserService userService;
    6. public void doController(){
    7. userService.doService();
    8. System.out.println("do Controller...");
    9. }
    10. }
    11. @Service
    12. public class UserService {
    13. public void doService(){
    14. System.out.println("do service....");
    15. }
    16. }
    17. @SpringBootApplication
    18. public class DemoApplication {
    19. public static void main(String[] args) {
    20. //Spring 上下文
    21. ApplicationContext context = SpringApplication.run(DemoApplication.class, args);
    22. //返回的就是 Spring 的运行环境
    23. UserController bean = context.getBean(UserController.class);
    24. bean.doController();
    25. }
    26. }

    运行结果如下 

    2.构造方法注入

    如果存只有一个构造函数,@Autowired 可以省略,当有多个构造函数的时候,默认使用无参的构造函数,如果没有无参的构造函数,我们要告诉 Spring 我们用哪个构造函数,在那个构造函数上面添加 @Autowired 就行

    1. package com.example.ioc.controller;
    2. import com.example.ioc.config.UserInfo;
    3. import com.example.ioc.service.UserService;
    4. import org.springframework.beans.factory.annotation.Autowired;
    5. import org.springframework.stereotype.Controller;
    6. @Controller
    7. //用 @Controller 告诉 Spring 帮我们管理这个对象
    8. public class UserController {
    9. //属性注入
    10. //@Autowired
    11. //private UserService userService;
    12. //构造方法注入
    13. private UserService userService;
    14. private UserInfo userInfo;
    15. // public UserController() {
    16. //
    17. // }
    18. public UserController(UserService userService) {
    19. this.userService = userService;
    20. }
    21. @Autowired
    22. public UserController(UserService userService, UserInfo userInfo) {
    23. this.userService = userService;
    24. this.userInfo = userInfo;
    25. }
    26. public void doController(){
    27. userService.doService();
    28. System.out.println("do Controller...");
    29. }
    30. }

    3.Setter 方法注入

    1. public class UserController {
    2. private UserService userService;
    3. @Autowired
    4. public void setUserService(UserService userService) {
    5. this.userService = userService;
    6. }
    7. public void doController(){
    8. userService.doService();
    9. System.out.println("do Controller...");
    10. }
    11. }

    当程序中同一个类型有多个对象时,使用@Autowired 会报错(一些情况下)

    解决方法如下:

    1,属性名和你需要使用的对象名保持一致

    2.使用@Primary 注解标识默认的对象,优先用它

    3.使用@Qualifier 

    4.使用@Resource

  • 相关阅读:
    Android Studio修改模拟器AVD Manger目录
    RTSP协议抓包及讲解
    HDLbits exercises 1 (开头到vector5节选题)
    Python 实现RGB和YUV互转
    [Redis]Redis客户端
    ROS与QT的那些事--0 该教程安排及后续安排
    MySQL 按年月日统计,创建视图
    深入理解Python中的Lambda(匿名函数)
    【先楫HPM6750系列】移植轻量级AI推理框架——TinyMaix
    Ubuntu 22 LTS 搭建SpringBoot服务器
  • 原文地址:https://blog.csdn.net/qq_40841463/article/details/134462266