【JavaEE进阶】——第四节.Spring更简单的实现Bean对象的存取(利用注解储存和注入Bean对象)

本文介绍了通过注解在Spring中存储和获取Bean对象的方法,包括@Service等五大类注解以及@Bean方法注解的使用。文章讨论了属性注入、setter注入和构造方法注入的优缺点,并提到了经典面试题,如各种注入方式的区别及@Resource与@Autowired的差异。
摘要由CSDN通过智能技术生成

作者简介:大家好,我是未央;

博客首页:未央.303

系列专栏:JavaEE进阶

每日一句:人的一生,可以有所作为的时机只有一次,那就是现在!!!

文章目录

前言

一、储存Bean对象

1.1 5大类注解

1.2 @Bean方法注解

二、获取Bean对象

2.1 属性注入

优缺点 

2.2 Setter注入

优缺点分析  

2.3 构造方法注入

 优缺点分析 

三、经典面试题

总结


前言

上一节内容我们学习了有关spring core的项目创建和使用,介绍了最基本的方法创建spring;

本节内容我们将介绍另外一种更加简单的方法,通过注解来注入bean对象;


上篇博客,我们讲了一个spring core项目的大致流程:

创建项目——》将对象储存到Spring容器中 ——》从Spring容器中取出Bean对象

但是吧,上篇讲的这些流程还是太繁琐了,有没有一个更简单的方式来实现对象的存取呢?

当然有,一起来看看吧!

首先,Spring项目的创建——这个没有什么好说的!就按我们上篇博客的步骤来进行。

但注意:与上篇博客相比,spring的配置文件发生了改变——为了更简单的实现Bean对象的存入(把对象存到spring中)

更改后的spring配置文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:content="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd">
    <content:component-scan base-package="在对象储存中,要扫描的路径"></content:component-scan>
</beans>

一、储存Bean对象

首先我们回忆下,我们执行储存Bean对象的方式。

之前我们存储 Bean 时,需要在 spring-config 中添加⼀⾏ bean 注册内容才⾏,如下图所示:

这种存入 Bean 的方式,并不是很好!

原因:

1、需要手动添加 bean 对象 到 配置文件中
2、如果 是配置文件中出现了问题,不好调试。
 

而现在我们不需要一个一个的在spring配置文件中添加我们要储存的Bean对象

我们直接:

 你还需要再类中添加注解——再扫描的过程中,只有添加了注解的类才能成功存储到spring中;

这就引起了我们注解的概念:

一共有两种方法:
(1)使用 5 大类注解实现

1、@Controller 【Controller - 控制器】
2、@Service 【service - 服务】
3、@Repository 【repository - 仓库】
4、@Configuration 【configuration - 配置/布局】
5、@Component 【component - 组件】

(2)通过 方法注解@Bean ,也可以将 一个 bean 存储到 Spring 中。


我们先介绍第一种方法:

1.1 5大类注解

@Service注解介绍:
 

上面我们用到了@Service注解,其他四个注解@Controller、@Repository也是一样的。 


补充1: 

既然这5大类注解的用法这么相似,那为啥还要分成5个不同的注解?统一弄成一个注解不好吗?

解答:

要解答这个问题,首先我们先要了解软件工程方面的知识;


之所以讲这个 关于“软件分层” 的定义呢,就是为了 后面讲解 “为什么要有五大注解类” 做铺垫!

既然这5大类注解对应着软件分层中的不同层——他们各自要实现的功能是不同的。

这也就不难理解为啥要分成5大类中注解了;


补充2:

你可能还有一个疑惑:为什么?我们的 Spring 一定要有下面这个配置呢?

解答:

如果没有这个配置,意味着什么??

大家可以想象一下:
一个 Spring 项目中,我们的类可分两种类型:

1、需要 进行 控制反转的类,将类的“生命周期”交给 Spring 来管理的类。【比如:UserController】

2、不需要存入 Spring 中的 类。

举例说明:

假设,我们有一个大型项目,需要存入 Spring 的 类 和 不需要存入 Spring 中的类,个数占比是五五开的。
这就会存在问题了!
如果我们没有 描述 根目录 的 这一行代码,
Spring 就会去扫描 所有的类,看看这些类中有哪些。
但是!项目中 需要存入 Spring 中的类,只占 50 %。
即:Spring 要浪费一倍的时间,去排查 不需要 存入 Spring 中的类。
所以,Spring 为了 提升效率,你必须要给我指定扫描的目录。
保证该目录下,一定是需要存入 Spring中的类。

这样 Spring就只需要扫描 对应目录中的类,就可以了!



1.2 @Bean方法注解

方法步骤:

上面我们通过5大类注解还有Bean方法实现了简单往spring中存Bean对象。那么怎样实现简单的从spring中取Bean对象呢?


二、获取Bean对象

获取 bean 对象也叫做 对象装配,是把对象取出来放到某个类中,有时候也叫 对象注入。

对象注入,其实就是我们spring中的DI——依赖注入

IoC 和 DI 是 Spring 中最重要的两个概念,其中 IoC(Inversion of Control)为控制反转的思想,而 DI(Dependency Injection)依赖注入为其(IoC)具体实现。那么 DI 实现依赖注入的方式有几种?这些注入方式又有什么不同?

在 Spring 中通过@Autowired实现依赖注入的常见方式有以下 3 种:

  1. 属性注入(Field Injection)
  2. Setter 注入(Setter Injection)
  3. 构造方法注入(Constructor Injection)

首先我们介绍第一个属性注入:

2.1 属性注入

因此,可以得出结论:

在使用 @Autowired 进行 属性注入的时候
如果 注入的对象,被多次 存入 Spring 中了,那么,光凭 属性的类型是找不到匹配的 bean的!需要将 属性的变量名 改成 BeanName,根据 BeanName 来找寻匹配的对象(bean)并进行 属性注入。

方法不止这一种,但是上述这种 “精确描述 bean 的名称” 的方法,是最简单高效的!

是不是很简单,不用去获取 Spring 的上下文对象 和 getBean 方法,直接通过一个注解,即可获取对应的bean(从Spring中取出 bean)。

举例说明:

当同一个类型的 Bean 存在多个时就有可能出现非唯一 Bean 的异常;

那么我们怎么解决这个问题呢?

两种办法:

 1、把你的属性名给改成类中的方法名student1或者student2;

2、用@Qualifier 是用来筛选 Bean 

优缺点 

优点:

属性注入最大的优点就是实现简单、使用简单只需要给变量上添加一个注解;(@Autowired),就可以在不 new 对象的情况下,直接获得注入的对象了(这就是 DI 的功能和魅力所在)

缺点:

  1. 功能性问题:无法注入一个不可变的对象(final 修饰的对象——无法被初始化);
  2. 通用性问题:只能适应于 IoC 容器;
  3. 设计原则问题:更容易违背单一设计原则(因为使用方便,在程序中可能大量使用属性注入)。

2.2 Setter注入


优缺点分析  

从上面代码可以看出,Setter 注入比属性注入要麻烦很多。

优点:

要说 Setter 注入有什么优点的话,那么首当其冲的就是它完全符合单一职责的设计原则,因为每一个 Setter 只针对一个对象

缺点:

但它的缺点也很明显,它的缺点主要体现在以下 2 点:

  1. 不能注入不可变对象(final 修饰的对象——无法通过构造方法初始化);
  2. 注入的对象可被修改(setter方法可多次被调用)

2.3 构造方法注入

 优缺点分析 

优点:

构造方法注入相比于前两种注入方法,它可以注入不可变对象,并且它只会执行一次,也不存在像 Setter 注入那样,被注入的对象随时被修改的情况,它的优点有以下 4 个:

可注入不可变对象(在构造方法中就进行了初始化);
注入对象不会被修改(一个类中,构造方法只能调用一次);
注入对象会被完全初始化;
通用性更好(即可以用于IoC容器,也可以用于非IoC容器)。

缺点:

因为在构造方法中可以传多个参数,容易违背单一性原则;



 

三、经典面试题

1.属性注入 ,构造方法注入 和 Setter 注入 之间,有什么区别?


2.@Resource VS @Autowired 的区别

@Autowired是spring提供的注解,@Resource是jdk提供的注解

1、用法不同

  • Autowired,支持 属性注入,构造方法注入,Setter 方法注入。
  • @Resource:支持 属性注入,Setter方法注入。不支持 构造方法注入。

2、@Resource 的 属性注入 比 @Autowired 的 属性注入,使用的更舒服。 因为 @Resource 有很多的属性可以设置;而 @Autowired 只有一个 value 属性。

有很多的属性,即意味着可以使用很多其它的功能。
比如在上面的当同一个类型的 Bean 存在多个时就有可能出现非唯一 Bean 的异常,如果是@Autowired,我们还需要额外的@Qualifier 是用来筛选并 Bean。

而@Resource则可以直接:


总结

今天的内容就分享到这里,我们下期再见!!!!!!

 

  • 8
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值