【JavaEE】Spring中注解的方式去获取Bean对象

【JavaEE】Spring的开发要点总结(3)

在这里插入图片描述

文章目录

  • 【JavaEE】Spring的开发要点总结(3)
    • 1. 属性注入
      • 1.1 @Autowired注解
      • 1.2 依赖查找 VS 依赖注入
      • 1.3 配合@Qualifier 筛选Bean对象
      • 1.4 属性注入的优缺点
    • 2. Setter注入
      • 2.1 @Autowired注解
      • 2.2 命名规则
      • 2.3 Setter注入的优缺点
    • 3. 构造方法注入
      • 3.1 @Autowired注解
      • 3.2 命名规则
      • 3.3 构造方法注入的优缺点
    • 4. 另一个注入可以用的注解@Resource
      • 4.1 来源不同
      • 4.2 匹配机制不同
      • 4.3 参数不同
      • 4.4 @Resource多一个匹配Bean对象名称的方案
      • 4.5 使用上的区别
    • 5. 综合练习

【JavaEE】Spring的开发要点总结(3)

在前面的代码里,我们获取Bean对象也比较麻烦:

在这里插入图片描述

本文章就是为了更方便地去获取Bean对象~

  • 对象装配
  • 也叫 对象注入

那么有没有对应的注解去实现这个功能呢?

Spring提供的三种实现方法:

  1. 属性注入
  2. 构造方法注入
  3. Setter注入

而这种非明文获取Bean对象的过程,就是DI

  • 而之前就是DL

首先,先创建一个新的项目,这次用规范的写法(工程分层):在这里插入图片描述

  • 配置文件自己去配~

在对应的层就是对应的注解:

在这里插入图片描述

1. 属性注入

在这里插入图片描述

这是数据持久层的一个插入方法,而 这个方法理论上是要在service层去调用的

在这里插入图片描述

但是new对象的写法在Spring已经不用啦:
在这里插入图片描述

属性注入则可以更便捷获取Bean对象~

1.1 @Autowired注解

在这里插入图片描述

自动接通吧,差不多那个意思~

利用这个注解,就能有以下操作:

在这里插入图片描述

在这里插入图片描述

这就是依赖注入,DI~

不用通过代码显示查找

  • 而是在Spring中,隐式用高效的方式自动地扫描到对应的Bean对象(n. 依赖)
  • 然后赋值(v. 注入)给这个成员属性

测试一下:

在这里插入图片描述

  • new对象的方式不行,跟Bean对象是否加载存储到Spring中五无关~

也可以通过单元测试的方式去测试代码,后期更新测试开发博客文章的时候讲解~

1.2 依赖查找 VS 依赖注入

参考之前的文章:

在这里插入图片描述

在这里体现出来的一点就是,依赖查找依赖Bean的名称,而依赖注入则会自动匹配和找到Bean对象,注入到属性中

  • 不是说依赖注入没有“查找”的过程,只是依赖注入重点在于注入

DI 流程是这样的:

  1. getType,从容器中获取对象
    • 如果能拿到唯一一个,那么就直接赋值给属性变量里
    • 这个时候变量名没有要求~
    • 正如上面写的~

在这里插入图片描述

  1. 如果找到多个,则根据名称去匹配对应的Bean对象!

在这里插入图片描述

测试结果:

在这里插入图片描述

  • 确实是不同的Bean对象~
  • 也验证了 DI 使用名称去匹配~
  • 这个时候你的变量名就必须是Bean对象其中一个的名称了~
    • 否则会报找不到,不唯一…的异常

1.3 配合@Qualifier 筛选Bean对象

你不管三七二十一,就想要一个响当当的属性名:“userSuperDao”,但是没有这个名的Bean对象(现在有多个Bean)

你就可以利用这个注解:

在这里插入图片描述

1.4 属性注入的优缺点

参考链接: https://juejin.cn/post/7135235294265081887

优点:实现/使用简单

缺点:

  1. 无法注入到一个不可变的变量(final 修饰的变量)

在这里插入图片描述

原因是final的特征:

  1. 被final修饰的成员变量要在
    • 定义时直接被赋值
    • 构造方法中第一次赋值
    • 实例代码块中第一次赋值
  2. 除了成员变量外的final变量,要在第一次赋值后不能被修改
  1. 通用性问题
    • 测试困难
    • 依赖项之间的耦合
    • 运行时的验证困难
    • 依赖关系的不清楚,复杂
    • 我们这种简单地用,@Autowired注解在属性上,去注入的方式可能不适用与非IoC容器的
      • 而Setter和构造方法注入是通过调用方法的,所以没有这个问题
  2. 设计原则问题:更容易违背单一设计原则

参考官方文档:Spring | Home

 单一设计原则要求一个类应该只有一个引起改变的原因,即一个类应该只有一个主要责任。当使用属性注入时,可能会出现以下情况:
  1. 因为开销比较小,在这个类里有啥需要就写一个属性,涉及多个职责耦合
  2. 依赖关系不清楚,乱,说到底还是属性写太多,设计不单一

但是,开发就是这样,不会很完美,3这个点很难避开

  • 有时候为了速度,牺牲耦合性的情况也有
    • 比如一个页面的信息有分类的,有不同的职责
    • 但是,如果访问这个页面,要去访问那么多个接口,就有很多次“三次握手四次挥手”了
  • 实际场景实际分析!

2. Setter注入

在这里插入图片描述

顾名思义,借此模仿,就是Setter方法加对应的注解

2.1 @Autowired注解

在这里插入图片描述

这就自动在Spring中找到Bean对象,然后通过Setter注入给对应的成员变量~

  • DI,是这样的

测试:

在这里插入图片描述

2.2 命名规则

命名规则跟属性注入基本一致,你可以理解为属性注入套了层皮,就是Setter注入

正好现在我有两个UserDao的Bean对象,来演示一下:

  • 一个aaa,一个userDao

在这里插入图片描述


在这里插入图片描述


在这里插入图片描述


在这里插入图片描述

观察一下,再看答案:

  1. 跟方法名无关
  2. 跟属性名无关
  3. 跟方法的参数名有关
  4. 可以结合@Qualifier注解

规则:

  1. 通过参数的类型,如果只获得一个Bean对象,那么这个属性的名字是啥都OK
  2. 如果获取到多个Bean对象,则需要 通过属性名进行匹配
    • @Qualifier去筛选,就不用考虑参数名的问题~

2.3 Setter注入的优缺点

优点:

  • 通常情况下,构造Setter只去Set一两个属性,并不会构造全部的Setter方法
  1. 所以Setter注入,更符合单一设计原则,就不会像属性注入那样的广泛

  2. 通用性更好一点(相对于属性注入)

    • 属性注入如果在别的容器上,失效了,但是Setter注入暴露出一个Set方法了呀,我们可以通过Set方法去挽救,自己手动赋值个Bean对象/new一个进去~
    • 而属性注入的属性,一般是private,在别的类不能直接修改其值

这个优点很牵强 ^ _ ^

缺点:

  1. 无法注入到一个final修饰的变量
  2. Setter注入的对象可以被修改
    • 相比于属性注入,Setter方法是公开的,所以这个方法可能被多次调用并修改
      • 这样就导致注入的结果被覆盖~
      • 或者原本设置的值被注入修改~
    • 属性注入也可以,但是是private,所以只能在对应的类中修改

3. 构造方法注入

这是官方推荐的一种注入方式~

  • 虽然如此,官方写的代码,用这个的不多🤣
  • 虽然更加完美,但是写起来麻烦

在这里插入图片描述

(Spring 4.x 之后推荐的注入方式)

3.1 @Autowired注解

在这里插入图片描述

  • 在构造方法上加上个@Autowired注解~

这也是标准的写法~

不标准的写法,不加@Autowired注解不会报错:

  • 因为官方推荐,所以搞了点小特殊🤣

在这里插入图片描述

  • 这个类如果只有一个构造方法的情况下,才是可以省略的!

在这里插入图片描述

  • 空的构造方法还行,就相当于啥也不注入,至少在构造UserService3的时候可以调用这个方法

在这里插入图片描述

  • 这个方法直接是不能调用了,因为不确定调用哪个构造方法,该注入啥

加上就不会有事:

在这里插入图片描述

原理就是:

  • 框架在构造UserService的时候,会根据@Autowried去挑选构造方法,而参数的来源就是Bean对象
  • 因此构造方法就会将这些Bean对象注入到属性上

所以不能有多个@Autowired修饰的构造方法:

在这里插入图片描述

3.2 命名规则

跟Setter注入一样!

  • 只不过不能用@Qualifier注解!

在这里插入图片描述

  1. 这个类只有一个Bean对象,那么参数名无所谓
  2. 这个类有多个Bean对象,那么参数名必须要匹配对应的Bean对象的名称

在这里插入图片描述

3.3 构造方法注入的优缺点

优点:

  1. 可以注入到一个final修饰的变量
    • 因为构造方法的第一次赋值final变量是允许的
    • 但是一定要保证每个构造方法涉及到的成员,都包含所有的final变量,即保证final变量都可以被初始化
      • 一个都不能缺

在这里插入图片描述

在这里插入图片描述

运行结果没问题:

在这里插入图片描述

  1. 注入的对象不会被修改,因为构造方法只会加载一次

    • 硬要修改还是可以的,只不过不能通过构造方法
  2. 保证使用的时候,注入对象都被初始化了

  3. 通用性更好(相对其他)

    • 同Setter注入
    • 因为必须传值,因为优点3

缺点:

  • 可以传入多个属性,不太符合单一设计原则
  • 这一点是写代码方面的初心问题吧。因为硬要这么说,你完全也可以给那个别的属性构造个Setter呀
  1. 写法比较复杂
  2. 无法解决循环依赖的问题
    • a依赖b,b依赖a,就是一个“循环”
    • 得用Spring三级缓存去解决
    • 这个循环依赖后面详细讲,现在就了解一下~

常见的面试题就是三者的区别:就是把他们的使用和优缺点讲一下~

4. 另一个注入可以用的注解@Resource

这个可以直接替换@Autowired注解,用IDEA专业版的话,有些场景下,用@Autowired是会报错的~

  • 但是属于误报,运行还是可以运行(设置把报错信息忽略掉)~

在这里插入图片描述

Resource就没啥问题:

在这里插入图片描述

用法基本一致:
在这里插入图片描述

这也是个常见的面试题:@Autowired和@Resource的区别是什么?

4.1 来源不同

@Autowired来自Spring框架,而@Resource来自JavaEE规范

4.2 匹配机制不同

  • @Autowired先进行类型匹配再用名称匹配
    • 但是专业版在这个注解判断代码正确与否的时候,并不会进一步用名称去匹配
    • 所以会误报
  • @Resource先进行名称匹配再进一步用类型匹配
    • 名称对不上,也会用类型去找,有多个Bean会报错,只有一个Bean就忽略名称的问题
  • 当然误报的原因可能有很多,甚至不是单一的原因…
  • IDEA兼容性不同~

4.3 参数不同

@Autowired只有一个参数,required,而@Resource没有这个参数,但是有很多其他的参数

在这里插入图片描述

required默认为true,代表要求Bean注入一定要存在

required设置为false,代表不要求Bean一定存在,不注入即可,就不会报找不到Bean对象异常~

在这里插入图片描述

在这里插入图片描述

4.4 @Resource多一个匹配Bean对象名称的方案

在这里插入图片描述

因此,@Resource多了一个匹配名称的方案:设置name的值即可:

在这里插入图片描述

4.5 使用上的区别

@Resource不支持构造方法注入

在这里插入图片描述

5. 综合练习

补充: 无论怎么样,静态属性是无法被注入Bean的,因为静态属性的加载是在Spring之前的,即类加载的时候,是这个类通用的属性,而Bean对象是一个实例!难不成每次获取Bean对象,Spring都会注入覆盖这个值吗~

这不合理~

  • 而依赖注入发生在你获取Bean对象的时候,Spring帮你构造实例的时候,而不是你自己new和设置的时候
  • 所以在main方法中,还是要通过DL去获取Bean

要求:在Spring项目中,通过main方法获取到Controller类,在Controller类里调用Service类,在Service类中获取Dao类,在Dao类中的一个方法,用伪代码new一个User,进行返回,返回给main方法,打印user。

在这里插入图片描述

在这里插入图片描述


在这里插入图片描述

报错了,这也是常见的问题,就Test类并没有在demo目录下,而是与其并列,要把他放进目录里:

在这里插入图片描述

运行结果:

在这里插入图片描述

是不是跟之前普通java代码对IoC的理解思想差不多😀

【JavaEE】JavaEE进阶:框架的学习 - Spring的初步认识_s:103的博客-CSDN博客


文章到此结束!谢谢观看
可以叫我 小马,我可能写的不好或者有错误,但是一起加油鸭🦆

代码链接:

  1. SpringDemo3 · 游离态/马拉圈2023年7月 - 码云 - 开源中国 (gitee.com)
  2. SpringDemo4 · 游离态/马拉圈2023年7月 - 码云 - 开源中国 (gitee.com)

用注解的方式去获取Bean对象,是不是很方便🤭


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/6276.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

21matlab数据分析牛顿插值(matlab程序)

1.简述 一、牛顿插值法原理 1.牛顿插值多项式   定义牛顿插值多项式为: N n ( x ) a 0 a 1 ( x − x 0 ) a 2 ( x − x 0 ) ( x − x 1 ) ⋯ a n ( x − x 0 ) ( x − x 1 ) ⋯ ( x − x n − 1 ) N_n\left(x\right)a_0a_1\left(x-x_0\right)a_2\left(x-x_0\…

SpringBoot 如何使用 EmbeddedDatabaseBuilder 进行数据库集成测试

SpringBoot 如何使用 EmbeddedDatabaseBuilder 进行数据库集成测试 在开发 SpringBoot 应用程序时,我们通常需要与数据库进行交互。为了确保我们的应用程序在生产环境中可以正常工作,我们需要进行数据库集成测试,以测试我们的应用程序是否能…

剑指offer61.扑克牌中的顺子

我的想法非常简单,就是先给数组排序,然后统计里面有几个0,然后遍历数组,如果是0或者比后面一个数小1就直接进入下一次循环,如果比后面一个数小2,就用掉一个0,0的数量减1,如果比后面的…

Pycharm----导入库文件夹不在py文件的目录下

问题描述: 想在不同目录下导入根目录的包,直接写会报错。如下边object_detect.py在function文件夹下,导入包默认在这个文件下,但我想导入根目录models和utils下的包 解决方法: 将根目录设置为源代码根目录&#xff0…

【OC总结 面向对象 + 内存管理 + runtime】

文章目录 前言面向对象1.1 一个NSObject对象占用多少内存?1.2 iOS的继承链 & 对象的指针指向了哪里?1.3 OC的类的信息存放在哪里?-isa指针1.4 isMemberOfClass & isKindOfClass Runtime1.4 讲一下OC的消息机制1.5 消息转发机制流程1.…

【指针和数组笔试题(1)】详解指针、数组笔试题

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言整型数组字符数组第一组题第二组题第三组题 总结 前言 在计算之前要了解基本概念: 数组名的理解 数组名是数组首元素的地址 有两个例外 1.sizeof(…

Linux网络基础 — 数据链路层

目录 数据链路层 认识以太网 局域网转发的原理 认识以太网的MAC报头 以太网帧格式 认识MAC地址 对比理解MAC地址和IP地址 基于MAC帧协议再次谈一谈局域网转发的原理 认识MTU MTU对IP协议的影响 MTU对UDP协议的影响 MTU对于TCP协议的影响 ARP协议 ARP协议的作用 …

Xcode 15 beta 4 (15A5195m) - Apple 平台 IDE

Xcode 15 beta 4 (15A5195m) - Apple 平台 IDE IDE for iOS/iPadOS/macOS/watchOS/tvOS/visonOS 请访问原文链接:https://sysin.org/blog/apple-xcode-15/,查看最新版。原创作品,转载请保留出处。 作者主页:sysin.org visonOS …

SpringBoot整合SpringCloudStream3.1+版本的Kafka死信队列

SpringBoot整合SpringCloudStream3.1版本的Kafka死信队列 上一篇直通车 SpringBoot整合SpringCloudStream3.1版本Kafka 实现死信队列步骤 添加死信队列配置文件,添加对应channel通道绑定配置对应的channel位置添加重试配置 结果 配置文件 Kafka基本配置&#…

C++ deque/queue/stack的底层原理

deque容器的存储结构 和 vector 容器采用连续的线性空间不同,deque 容器存储数据的空间是由一段一段等长的连续空间构成,各段空间之间并不一定是连续的,可以位于在内存的不同区域。 deque采用一块所谓的map数组(注意&#xff0c…

rabbitmq模块启动报java.net.SocketException: socket closed的解决方法

问题 最近在接手一个项目时,使用的是spring-cloud微服务构架,mq消息消费模块是单独一个模块,但启动这个模块一直报如下错误: java.net.SocketException: socket closed 这个错误是这个模块注册不到nacos报的错,刚开…

day34-Animated Countdown(动画倒计时)

50 天学习 50 个项目 - HTMLCSS and JavaScript day34-Animated Countdown&#xff08;动画倒计时&#xff09; 效果 index.html <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta name"viewport&q…

态势标绘专题介绍

介绍 这个专栏是专门针对基于Cesium来实现态势标绘的专题专栏,专栏主要实现了30余种态势几何形状的标绘和编辑、文本的标绘和编辑、图片的标绘和编辑以及简单模型的标绘,同时支持标绘结果的导出以及导入。包括最终编写成的一个完整的Vue3.2+TS+Cesium1.107.2的标绘组件。专栏…

从用户的角度谈GPT时代技术突破的两大关键逻辑

大家好,我是herosunly。985院校硕士毕业,现担任算法研究员一职,热衷于机器学习算法研究与应用。曾获得阿里云天池比赛第一名,CCF比赛第二名,科大讯飞比赛第三名。拥有多项发明专利。对机器学习和深度学习拥有自己独到的见解。曾经辅导过若干个非计算机专业的学生进入到算法…

EtherCAT转TCP/IP网关EtherCAT解决方案

你是否曾经为生产管理系统的数据互联互通问题烦恼过&#xff1f;曾经因为协议不同导致通讯问题而感到困惑&#xff1f;现在&#xff0c;我们迎来了突破性的进展&#xff01; 介绍捷米特JM-TCPIP-ECT&#xff0c;一款自主研发的Ethercat从站功能的通讯网关。它能够连接到Etherc…

通过FPGA实现基于RS232串口的指令发送并控制显示器中目标位置

目录 1.算法理论概述 串口通信模块 指令解析模块 位置控制模块 显示器驱动模块 2.部分核心程序 3.算法运行软件版本 4.算法运行效果图预览 5.算法完整程序工程 1.算法理论概述 通过FPGA实现基于RS232串口的指令发送并控制显示器中目标位置是一种常见的应用场景&#x…

Prompt 技巧指南-让 ChatGPT 回答更准确

随着 ChatGPT 等大型语言模型 (LLM)的兴起&#xff0c;人们慢慢发现&#xff0c;怎么样向 LLM 提问、以什么技巧提问&#xff0c;是获得更加准确的回答的关键&#xff0c;也由此产生了提示工程这个全新的领域。 提示工程(prompt engineering)是一门相对较新的领域&#xff0c;用…

java学习003

Java数组 Java 语言中提供的数组是用来存储固定大小的同类型元素&#xff0c;这一点和PHP语言的可变数组长度不同。 声明变量数组 首先必须声明数组变量&#xff0c;才能在程序中使用数组。下面是声明数组变量的语法&#xff1a; dataType[] arrayRefVar; // 首选的方法 或 …

云计算——云计算与虚拟化的关系

作者简介&#xff1a;一名云计算网络运维人员、每天分享网络与运维的技术与干货。 座右铭&#xff1a;低头赶路&#xff0c;敬事如仪 个人主页&#xff1a;网络豆的主页​​​​​ 目录 前言 一.虚拟化 1.什么是虚拟化 2.虚拟化技术作用 二.云计算与虚拟化的关系 三.虚…

华为eNSP:ospf的配置

一、拓扑图 二、路由器的配置 1、路由器依据规划配置接口IP AR1: <Huawei>system-view [Huawei]int g0/0/0 [Huawei-GigabitEthernet0/0/0]ip add 10.10.10.1 24 [Huawei-GigabitEthernet0/0/0]qu AR2: <Huawei>system-view [Huawei]int g0/0/0 [Huawei-Gi…