[JavaEE] Spring IoC&DI

目录

一、IoC

[1.1 IoC 概念](#1.1 IoC 概念)

[1.2 IoC的作用](#1.2 IoC的作用)

二、DI

[2.1 DI 概念](#2.1 DI 概念)

[2.2 属性注入(Field Injection)](#2.2 属性注入(Field Injection))

[2.3 构造方法注入(Constructor Injection)](#2.3 构造方法注入(Constructor Injection))

[2.4 Setter 注入(Setter Injection)](#2.4 Setter 注入(Setter Injection))

[2.5 三种注入优缺点分析](#2.5 三种注入优缺点分析)

[2.5.1 属性注入](#2.5.1 属性注入)

[2.5.2 构造函数注入](#2.5.2 构造函数注入)

[2.5.3 Setter 注入](#2.5.3 Setter 注入)

[三、IoC & DI 使用](#三、IoC & DI 使用)

[四、Bean 的存储](#四、Bean 的存储)

[4.1 @Controller(控制器存储)](#4.1 @Controller(控制器存储))

[4.2 @Service(服务存储)](#4.2 @Service(服务存储))

[4.3 @Repository(仓库存储)](#4.3 @Repository(仓库存储))

[4.4 @Component(组件存储)](#4.4 @Component(组件存储))

[4.5 @Configuration(配置存储)](#4.5 @Configuration(配置存储))

[4.6 Bean 的获取](#4.6 Bean 的获取)

[4.6.1 根据类名称获取 bean](#4.6.1 根据类名称获取 bean)

[4.6.2 根据名称获取 bean](#4.6.2 根据名称获取 bean)

[4.6.3 根据类名和名称获取 bean](#4.6.3 根据类名和名称获取 bean)

[五、方法注解 @Bean](#五、方法注解 @Bean)

[5.1 同一类多个对象](#5.1 同一类多个对象)

[5.2 重命名 Bean](#5.2 重命名 Bean)

[5.3 @Autowired 存在的问题](#5.3 @Autowired 存在的问题)

[5.4 @Autowired 和 @Resource 的区别](#5.4 @Autowired 和 @Resource 的区别)

[5.5 Spring Frameworke Spring Boot Spring MVC 的关系和区别](#5.5 Spring Frameworke Spring Boot Spring MVC 的关系和区别)

六、Bean的命名

[6.1 五大注解](#6.1 五大注解)

[6.2 @Bean](#6.2 @Bean)


一、IoC

1.1 IoC 概念

IoC: Inversion of Control(控制反转), 指的是获得依赖对象的过程被反转了 , 也就是当需要某个对象时, 传统开发模式中需要自己通过 new 创建对象, 现在把创造对象的任务交给容器 , 程序中只需要依赖注入(Dependency Injection,DI)。我们把这个容器成为: IoC 容器, Spring 是一个 IoC 容器, 所以有时 Spring 也称为 Spring 容器。

1.2 IoC的作用

现实开发思路: 先设计轮子,然后根据轮子大小设计底盘, 根据底盘设计车身, 最后根据车身设计整个汽车。这里就出现了"依赖" 关系, 汽车依赖车身, 车身依赖底盘, 底盘依赖轮子。

这样的思路当我们修改底层的时候, 整个调用链上的所有代码都需要修改, 耦合度非常高。所以我们尝试先设计汽车车身, 根据车身来设计底盘, 这样依赖关系就倒置过来了。在代码中实现时, 我们只需要将原来自己创建的下级类, 改为传递的方式(也就是注入的方式), 这样就完成了程序的解耦合, 使用 IoC 之后控制权发生反转, 不再是使用方对象创建并控制依赖对象了, 而是把依赖对象注入到当前对象中, 依赖对象的控制权不再由当前类控制了。

IoC 优势

1.、资源集中管理:IoC容器会帮我们管理一些资源(对象等), 我们需要使用时, 只需要从IoC容器中去取就可以了。

2、我们在创建实例时不需要了解其中的细节, 降低了使用资源双方的依赖程度, 也就是耦合度。

二、DI

2.1 DI 概念

DI: Dependency Injection(依赖注⼊), 容器在运行期间, 动态的为应用程序提供运行时所依赖的资源, 称之为依赖注入。从这点来看, DI 和 IoC 是从不同角度描述同一件事情, DI 是从应用程序的角度来描述, 通过引入 IoC 容器, 程序利用依赖关系注入的方式, 实现对象之间的解耦合。 简单来说, IoC 是现象, DI 是 IoC 的实现方式。

我们可以使用**@Autowired** 注解 , 完成依赖注入的操作, 简单来说就是把对象 取出来放到某个类的属性中, 关于依赖注入, Spring 提供了三种方式:

2.2 属性注入(Field Injection)

属性注入是使用 @Autowired 实现的, 将 Service 类注入到 Controller 类中, 下方是 Service 类代码

java 复制代码
@Service
public class UserService {
    @Autowired
    public UserDao userDao;
    public void say(){
        System.out.println("UserService say......");
    }
}

Controller类代码如下:

java 复制代码
@RestController
@RequestMapping("/user")
public class UserController {
    @Autowired
    private UserService userService;
    @RequestMapping("/say")
    public void say(){
        System.out.println("UserController say......");
        userService.say();
    }
}

获取 Controller 中的 say 方法:

java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserController userController = (UserController) context.getBean("userController");
        userController.say();
    }
}

运行结果如下:

2.3 构造方法注入(Constructor Injection)

构造方法注入就是在类的构造方法中实现注入, 如下方代码所示:

java 复制代码
@RestController
@RequestMapping
public class UserController2 {
    private UserService userService;
    public UserController2() {
    }
    @Autowired
    public UserController2(UserService userService) {
        this.userService = userService;
    }
    public void say(){
        System.out.println("UserController2 say.......");
        userService.say();
    }
}

如果只有一个构造方法, 那么 @Autowired 可以省略, 但是如果有多个构造方法, 那需要加上 @Autowired 来明确指定使用哪个构造方法, 并且如果不写 @Autowired 是系统生成的默认的构造方法。

2.4 Setter 注入(Setter Injection)

Setter 注入和属性的 Setter 方法实现类似, 只不过在设置 set 方法的时候需要加上 @Autowired 注解, 并且 Spring 会自动调用方法来帮助数据初始化, 如下方代码所示:

java 复制代码
@RestController
@RequestMapping("/user")
public class UserController {

    private UserService userService;
    @RequestMapping("/say")
    @Autowired
    public void SetUserService(UserService userService) {
        this.userService = userService;
    }
    public void say(){
        System.out.println("UserController say......");
        userService.say();
    }
}

2.5 三种注入优缺点分析

2.5.1 属性注入

2.5.1.1 优点:简洁, 使用方便

2.5.1.2 缺点:由于 @Autowired 来自于 Spring , 只能使用于 IoC 容器, 如果是非 IoC 容器不可用, 并且只有在使用时才会出现空指针异常;并且不能注入一个 Final 修饰的属性, 除非 final 修饰的属性被构造方法初始化, 或者给 final 修饰的属性初始化。(效率不如构造方法注入)

2.5.2 构造函数注入

2.5.2.1 优点:可以注入 final 修饰的属性, 注入的对象不会被修改, 通用性好, **构造方法(不是构造函数注入)**是JDK支持的, 任何框架都适用。 依赖对象在使用前一定会被完全初始化, 因为依赖在类的构造方法中执行, 而构造方法是在类加载阶段就会执行。

2.5.2.2 缺点:注入多个对象时会比较繁琐(每个参数都需要在构造方法手动写入)。

2.5.3 Setter 注入

2.5.3.1 优点:方便在类实例后, 重新对该对象配置或者注入(可以简单的理解为调用方法)

2.5.3.2 缺点:不能注入 final 修饰的属性, 注入对象可能会被改变, 因为 setter 方法可能会被多次调用。

三、IoC & DI 使用

Spring 是一个 IoC 容器, 作为容器就具备存储功能, Spring 容器主要管理的是对象, 这些对象我们称之为 "Bean" , 我们可以通过注解, 把这些对象交给 Spring 管理, 并且可以使用 @Autowired 来实现依赖注入(也就是当需要当前对象时, 从 Spring IoC 容器中自动拿取)。

  1. 把 UserDao 交给 Spring 管理, 由 Spring 来管理并创建类对象
java 复制代码
@Component
public class UserDao {
    public void say(){
        System.out.println("UserDao say......");
    }
}
  1. 把 UserService 交给 Spring 管理
java 复制代码
@Service
public class UserService {
    @Autowired
    public UserDao userDao;
    public void say(){
        System.out.println("UserService say......");
        userDao.say();
    }
}
  1. 由 UserController 开启调用, 并从 Spring 获取对象
java 复制代码
@RestController
@RequestMapping("/user")
public class UserController {

    @Autowired
    private UserService userService;
    @RequestMapping("/say")
    public void say(){
        System.out.println("UserController say......");
        userService.say();
    }

}

运行结果如下:

四、Bean 的存储

前面我们提到 IoC 控制反转, 就是将对象的控制权交给 Spring 的 IoC 容器, 由 IOC 容器创建及管理对象, 也就是 Bean 的存储。

共有两类注解可以实现:

类注解: @Controller、@Service、@Repository、@Component、@Configuration

方法注解: @Bean

4.1 @Controller(控制器存储)

使用@Controller 存储 bean 代码如下:

java 复制代码
@Controller // 将对象存储到 Spring 中 
public class UserController {
    public void sayHi(){
        System.out.println("hi,UserController...");
    }
}

通过 ApplicationContext 来接收启动类返回的 Spring 的 IOC 容器:

复制代码
ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);

通过 context.getBean 来获取类对象:

复制代码
UserController userController = context.getBean(UserController.class);
java 复制代码
@SpringBootApplication
public class SpringIocDemoApplication {
    public static void main(String[] args) {
        //获取Spring上下⽂对象 
        ApplicationContext context =
                SpringApplication.run(SpringIocDemoApplication.class, args);
        //从Spring上下⽂中获取对象 
        UserController userController = context.getBean(UserController.class);
        //使⽤对象 
        userController.sayHi();
    }
}

4.2 @Service(服务存储)

存储 bean 代码如下:

java 复制代码
@Service
public class UserService {
    public void sayHi(String name) {
        System.out.println("Hi," + name);
    }
}

读取 bean代码:

java 复制代码
@SpringBootApplication
public class SpringIocDemoApplication {
    public static void main(String[] args) {
        //获取Spring上下⽂对象 
        ApplicationContext context =
                SpringApplication.run(SpringIocDemoApplication.class, args);
        //从Spring中获取UserService对象 
        UserService userService = context.getBean(UserService.class);
        //使⽤对象 
        userService.sayHi();
    }
}

4.3 @Repository(仓库存储)

存储 bean 代码如下:

java 复制代码
@Repository
public class UserRepository {
    public void sayHi() {
        System.out.println("Hi, UserRepository~");
    }
}

读取 bean 代码:

java 复制代码
@SpringBootApplication
public class SpringIocDemoApplication {
    public static void main(String[] args) {
        //获取Spring上下⽂对象 
        ApplicationContext context =
                SpringApplication.run(SpringIocDemoApplication.class, args);
        //从Spring上下⽂中获取对象 
        UserRepository userRepository = context.getBean(UserRepository.class);
        //使⽤对象 
        userRepository.sayHi();
    }
}

4.4 @Component(组件存储)

存储 bean 代码如下:

java 复制代码
@Component
public class UserComponent {
    public void sayHi() {
        System.out.println("Hi, UserComponent~");
    }
}

读取 bean 的代码:

java 复制代码
@SpringBootApplication
public class SpringIocDemoApplication {
    public static void main(String[] args) {
        //获取Spring上下⽂对象 
        ApplicationContext context =
                SpringApplication.run(SpringIocDemoApplication.class, args);
        //从Spring上下⽂中获取对象 
        UserComponent userComponent = context.getBean(UserComponent.class);
        //使⽤对象 
        userComponent.sayHi();
    }
}

4.5 @Configuration(配置存储)

存储 bean 代码如下:

java 复制代码
@Configuration
public class UserConfiguration {
    public void sayHi() {
        System.out.println("Hi,UserConfiguration~");
    }
}

读取 bean 代码如下

java 复制代码
@SpringBootApplicationpublic class SpringIocDemoApplication {
    public static void main(String[] args) {
        //获取Spring上下⽂对象 
        ApplicationContext context =
                SpringApplication.run(SpringIocDemoApplication.class, args);
        //从Spring上下⽂中获取对象 
        UserConfiguration userConfiguration =
                context.getBean(UserConfiguration.class);
        //使⽤对象 
        userConfiguration.sayHi();
    }
}

上述所提到的多个类注解, 是为了让程序员看到类注解后, 就能直接了解当前类的用途

@Controller: 控制层, 接受请求, 对请求进行处理并响应

@Servie: 业务逻辑层, 处理具体的业务逻辑

@Repository: 数据访问层, 也称为持久层, 负责数据访问操作

@Configuration: 配置层, 处理项目中的一些配置信息

Config 多定义提供 bean, Component 多依赖@Autowired 注入 bean(多使用依赖注入)。

4.6 Bean 的获取

通过不同的方法我们拿到的对象可以是一个, 这个就是单例模式的体现。

4.6.1 根据类名称获取 bean
java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserController bean = context.getBean(UserController.class);
        bean.say();
    }
}
4.6.2 根据名称获取 bean
java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserController bean2 = (UserController) context.getBean("userController");
        bean2.say();
    }
}
4.6.3 根据类名和名称获取 bean

当一个类中有多个 Bean 的时候需要用到这个方法

java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserController bean3 = (UserController) context.getBean("userController", UserController.class);
        bean3.say();
    }
}

五、方法注解 @Bean

使用外部包里面的类, 没办法添加类注解, 或者当我们一个类, 需要多个对象的时候, 我们就要用到方法注解@Bean

5.1 同一类多个对象

如下代码所示:

java 复制代码
@Component
public class BeanConfig {
    @Bean
    public UserInfo user1() {
        UserInfo userInfo = new UserInfo();
        userInfo.setAge(18);
        userInfo.setName("zhangsan");
        return userInfo;
    }
    @Bean
    public UserInfo user() {
        UserInfo userInfo = new UserInfo();
        userInfo.setAge(16);
        userInfo.setName("lisi");
        return userInfo;
    }
}

如果一个类存在多个对象时我们使用类名调用Bean则会报错

java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserInfo userInfo = (UserInfo) context.getBean(UserInfo.class);
    }
}

通过 Bean 名称获取同一个类里多个对象的时候则不会报错

java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserInfo userInfo1 = (UserInfo) context.getBean("user1");
        UserInfo userInfo2 = (UserInfo) context.getBean("user");
        System.out.println(userInfo1);
        System.out.println(userInfo2);
    }
}

5.2 重命名 Bean

我们可以通过 name 属性给 Bean 对象进行重命名操作, 代码如下:

java 复制代码
@Bean(name = {"user1","u1"})
public UserInfo user1() {
    UserInfo userInfo = new UserInfo();
    userInfo.setAge(18);
    userInfo.setName("zhangsan");
    return userInfo;
}

我们使用 u1 就可以获取 User 对象, 代码如下所示

java 复制代码
@SpringBootApplication
public class Demo1Application {
    public static void main(String[] args) {
        ApplicationContext context =  SpringApplication.run(Demo1Application.class, args);
        UserInfo userInfo = (UserInfo) context.getBean("u1");
        System.out.println(userInfo);
    }
}

其中 name = {} 可以省略 , 代码如下:

java 复制代码
@Bean({"user1","u1"})
public UserInfo user1() {
    UserInfo userInfo = new UserInfo();
    userInfo.setAge(18);
    userInfo.setName("zhangsan");
    return userInfo;
}

并且只有一个名称的时候 {} 也可以省略, 代码如下:

java 复制代码
@Bean("user1")
public UserInfo user1() {
    UserInfo userInfo = new UserInfo();
    userInfo.setAge(18);
    userInfo.setName("zhangsan");
    return userInfo;
}

我们前面使用的注解声明 bean 不一定都会生效, bean 想要生效, 还需要被 Spring 扫描到, 我们需要把启动类放在我们希望扫描的包路径下, 这样我们定义的 bean 才可以被扫描到。

其中我们可以手动修改启动类的扫描路径, 只需要在启动类里加 @ComponentScan({""}) , 就可以自定义扫描路径, 而我们即使不加 @ComponentScan({""}) , 启动类声明注解也会包含。

5.3 @Autowired 存在的问题

注入时一个类里有多个 @Bean 时, 按照方法名匹配。

注入时一个类里有多个 @Bean 时, 并且与所有方法名称都不匹配, 启动会报错

解决方法:1. 在 @Bean 上加 @Primary(从依赖注入出发, 被调用方)

  1. 在 @Autowired 上加 @Qualifier, 或者在参数上加@Qualifier(从 IoC 角度出发, 主动调用方),并且 @Qualifier 优先级大于@Primary,并且@Qualifier优先级最高

  2. @Resource(name = " ") (JDK 原生支持的)

如果类只有一个 @Bean, 即使方法名称不匹配, 也会自动调用

5.4 @Autowired 和 @Resource 的区别

Autowired 根据类型匹配 Resource 根据名称匹配

Autowired 根据类型匹配, 但是 Autowired 后不能加名称, Autowired 是 Spring 框架提供的注解

优先按照类型匹配, 同类有多个对象按照名称匹配

Resource 根据名称匹配, Resource 能加名称, Resource 是 Spring 框架提供的注解

5.5 Spring Frameworke Spring Boot Spring MVC 的关系和区别

在我们创建 SpringBoot 项目的时候, 勾选的 Spring Web 框架其实就是 Spring MVC 框架。SpringBoot 是实现 Spring MVC 的其中一种方式, SpringBoot 可以添加很多依赖, 借助这些依赖来实现不同的功能, SpringBoot 通过添加 Spring Web MVC 框架来实现web功能。

六、Bean的命名

6.1 五大注解

(1) 前两位单词首字母均为大写, bean 名称为类名

(2) 不符合条件一的为类名首字母小写

(3) 可以通过 value 属性设置 @Controller(value = "user" ) 来自定义类名

6.2 @Bean

(1) bean 名称为方法名

(2) 可以通过 name 属性设置 @Bean(name = {"u1","user1"}) 来自定义名称

相关推荐
苹果酱05671 小时前
Golang的文件加密技术研究与应用
java·vue.js·spring boot·mysql·课程设计
xweiran3 小时前
CAS操作的底层原理(总线锁定机制和缓存锁定机制 )
java·cas·处理器·总线锁定·缓存锁定
V+zmm101343 小时前
基于微信小程序的水果销售系统的设计与实现springboot+论文源码调试讲解
java·微信小程序·小程序·毕业设计·springboot
头发那是一根不剩了3 小时前
java.lang.NoClassDefFoundError: javax/xml/bind/DatatypeConverter
java
小白起 v4 小时前
三天学完微服务其二
java·微服务·架构
huiyunfei4 小时前
MinorGC FullGC
java·jvm·算法
XWM_Web4 小时前
JavaAPI.02.包装类与正则表达式
java·开发语言·学习·eclipse
PangPiLoLo4 小时前
架构学习——互联网常用架构模板
java·学习·微服务·云原生·架构·系统架构·nosql
!!!5254 小时前
SpringBoot-web入门程序剖析
java·spring boot·后端