[JavaEE] Spring IoCDI

embedded/2025/1/13 9:36:03/

目录

一、IoC

1.1 IoC 概念

1.2 IoC的作用

二、DI

2.1 DI 概念

2.2 属性注入(Field Injection)

2.3 构造方法注入(Constructor Injection)

2.4 Setter 注入(Setter Injection)

2.5 三种注入优缺点分析

2.5.1 属性注入

2.5.2 构造函数注入

2.5.3 Setter 注入

三、IoC & DI 使用

四、Bean 的存储

4.1 @Controller(控制器存储)

4.2 @Service(服务存储)

4.3 @Repository(仓库存储)

4.4 @Component(组件存储)

4.5 @Configuration(配置存储)

4.6 Bean 的获取

4.6.1 根据类名称获取 bean

4.6.2 根据名称获取 bean

4.6.3 根据类名和名称获取 bean

五、方法注解 @Bean

5.1 同一类多个对象

5.2 重命名 Bean

5.3 @Autowired 存在的问题

5.4 @Autowired 和 @Resource 的区别

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

六、Bean的命名

6.1 五大注解

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 {@Autowiredpublic UserDao userDao;public void say(){System.out.println("UserService say......");}
}

Controller类代码如下: 

java">@RestController
@RequestMapping("/user")
public class UserController {@Autowiredprivate 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() {}@Autowiredpublic 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")@Autowiredpublic 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......");}
}

2. 把 UserService 交给 Spring 管理

java">@Service
public class UserService {@Autowiredpublic UserDao userDao;public void say(){System.out.println("UserService say......");userDao.say();}
}

3. 由 UserController 开启调用, 并从 Spring 获取对象

java">@RestController
@RequestMapping("/user")
public class UserController {@Autowiredprivate 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 {@Beanpublic UserInfo user1() {UserInfo userInfo = new UserInfo();userInfo.setAge(18);userInfo.setName("zhangsan");return userInfo;}@Beanpublic 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(从依赖注入出发, 被调用方)

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

                            3. @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"}) 来自定义名称


http://www.ppmy.cn/embedded/153518.html

相关文章

【conda】迁移到其他ubuntu机器

想把部署在一台4070机器的conda 环境导出到3090机器查看已安装的环境 conda env list环境导出 conda env export > semantic-text2image-search.yml 4070的conda环境导出 nvidia-smi 命令可以查看 GPU 的基本信息,包括显存、CUDA 驱动版本等: Driver Version:当前 NVIDI…

ros2笔记-6.2 使用urdf创建机器人模型

本节主要跟着小鱼老师的视频操作,不同的仿真平台有不同的建模语言,但是几乎都支持URDF。 本节使用URDF创建一个机器人模型。 6.2.1 帮机器人创建一个身体 URDF使用XML来描述机器人的结构和传感器、执行器等信息。 在chapt6/chap6_ws/src创建功能包:r…

GCC安全编译选项

文章目录 GCC安全编译选项1.BIND_NOW2.NX3.PIC4.PIE5.SP6.NO Rpath/Runpath7.FS8.Ftrapv9.Strip10.安全编译Python GCC安全编译选项 1.BIND_NOW 定义:立即绑定,打开GOT表重定位只读选项 使用方式:LDFlAGS"-Wl,-z,relro,-z,now" ./configure 2.NX 堆栈不可执行:打…

Golang笔记——rune和byte

大家好,这里是Good Note,关注 公主号:Goodnote,专栏文章私信限时Free。本文详细介绍Golang中的两种字符类型rune和byte,介绍他们的区别,编码方式和简单的使用。 文章目录 byte 类型rune 类型UTF-8 与 Unico…

2024最新版Node.js下载安装保姆级教程【图文详解】

Node.js 是一个基于 Chrome V8 引擎的 JavaScript 运行时环境,允许开发者在服务器端运行 JavaScript。它以非阻塞、事件驱动的架构处理高并发,适用于构建高效的网络应用。在 APP 自动化测试中,Node.js 常用于构建测试框架(如 WebD…

开源临床试验软件OpenClinica的安装

本文是为帮网友 A萤火虫 解决安装问题做的记录; 简介 什么是 OpenClinica ? OpenClinica 是世界上第一个商业开源临床试验软件,主要用于电子数据捕获(EDC)和临床数据管理(CDM)。它的设计旨在优…

【简博士统计学习方法】第2章:3. 感知机——学习算法之原始形式:算法解说

3. 感知机——学习算法之原始形式:算法解说 3.1 学习问题 给定训练数据集: T { ( x 1 , y 1 ) , ( x 2 , y 2 ) ⋯ , ( x N , y N ) } T\left\{\left(x_{1}, y_{1}\right),\left(x_{2}, y_{2}\right) \cdots,\left(x_{N}, y_{N}\right)\right\} T{(x…

LLM - Llama 3 的 Pre/Post Training 阶段 Loss 以及 logits 和 logps 概念

欢迎关注我的CSDN:https://spike.blog.csdn.net/ 本文地址:https://spike.blog.csdn.net/article/details/145056912 Llama 3 是 Meta 公司发布的开源大型语言模型,包括具有 80 亿和 700 亿参数的预训练和指令微调的语言模型,支持…