【Spring】依赖注入(DI)时常用的注解@Autowired和@Value

news2024/11/29 12:31:24

目录

1、@Autowired 自动装配

1.1、要实现自动装配不是一定要使用@Autowired

 1.2、@Autowired的特性

(1)首先会根据类型去spring容器中找(bytype),如果有多个类型,会根据名字再去spring容器中找(byname)

(2)如果根据名字还是匹配不到,解决方案为下面两个:

(3)可以写在方法、构造函数、字段、参数上

2、@Value

(1)方式一:直接把值写在@Value中的属性里

(2)方式二:对外部属性(SpringBoot配置文件)文件的引用

(3)SPEL表达式(Spring Expression Language):@Value("#{}")

 

        依赖注入(DI)是一种软件设计模式,旨在降低模块之间的耦合度。其核心思想是将对象之间的依赖关系外部化,使得代码更加灵活、可维护和可测试。在DI模式中,对象不再负责创建它所依赖的对象,而是由外部的容器(通常是框架或容器)负责将依赖的对象注入到目标对象中。框架如Spring提供了DI功能,通过注解、XML配置文件或Java配置类等方式,开发者可以在需要注入的地方指定相应的注解或配置,框架即可自动完成依赖对象的注入,无需手动创建和管理。

1、@Autowired 自动装配

1.1、要实现自动装配不是一定要使用@Autowired

(1)@Bean 方法的参数会自动注入;

(2)构造函数上的参数会自动注入

 1.2、@Autowired的特性

(1)首先会根据类型去spring容器中找(bytype),如果有多个类型,会根据名字再去spring容器中找(byname)

上图中涉及的代码,先看这个代码,后面作为示例会有改动

@Component  //这里配置bean的类型是AutowiredTestService,名字是autowiredTestService
public class AutowiredTestService {
}


@Component
public class AutowiredService {
    //首先会根据类型去spring容器中找(bytype) ,如果有多个类型, 会根据名字再去spring容器中找(byname)
    //这里会根据类型找到两个bean,一个是通过@Component注解配置的AutowiredTestService类的bean
    //还有一个是在SpringConfig配置类中通过@Bean注解配置的AutowiredTestService类的bean
    //然后他再会通过这里注入时的名字:autowiredTestService,
    //找到通过@Component注解配置的bean(autowiredTestService)
    @Autowired  //这里注入的类型是AutowiredTestService,名字是autowiredTestService
    AutowiredTestService autowiredTestService;
    @Override
    public String toString() {
        return "AutowiredService{" +
        "autowiredTestService=" + autowiredTestService +
        '}';
    }
}


@Configuration
public class SpringConfig {

    @Bean  //这里配置bean的类型是AutowiredTestService,名字是autowiredTestService2
    public AutowiredTestService autowiredTestService2(){
        AutowiredTestService autowiredTestService = new AutowiredTestService();
        autowiredTestService.setName("@Bean"); //为AutowiredTestService类bean对象中的name属性赋值
        return  autowiredTestService;
    }
}


@SpringBootTest  //需要加@SpringBootTest注解,否则这里需要自己去获取ioc容器才能从ioc容器中拿bean
public class AutowiredTest {
    @Autowired  
    AutowiredService autowiredService;

    @Test
    public void test01(){
        System.out.println(autowiredService.toString());
    }
}

运行结果:因为通过@Component注解配置的AutowiredTestService类bean时,没有为AutowiredTestService类bean对象中的name属性赋值,这里的结果中name等于null,所以这里获取的是通过@Component注解配置的AutowiredTestService类bean

把AutowiredService类中注入的AutowiredTestService的bean名字改为autowiredTestService2

@Component
public class AutowiredService {
    //首先会根据类型去spring容器中找(bytype) ,如果有多个类型, 会根据名字再去spring容器中找(byname)
    //这里会根据类型找到两个bean,一个是通过@Component注解配置的bean,名字为autowiredTestService
    //还有一个是在SpringConfig配置类中通过@Bean注解配置的AutowiredTestService类的bean
    //然后他再会通过这里注入时的名字:autowiredTestService2,
    //找到通过@Bean注解配置的bean,并且这个Bean对象中的name属性值为‘@Bean’
    @Autowired  //这里注入的类型是AutowiredTestService,名字是autowiredTestService2
    AutowiredTestService autowiredTestService2;
    @Override
    public String toString() {
        return "AutowiredService{" +
        "autowiredTestService=" + autowiredTestService2 +
        '}';
    }
}

运行结果:因为通过@Bean注解配置bean时,为AutowiredTestService类bean对象中的name属性赋值为‘@Bean’,这里的结果中name等于‘@Bean’,所以这里获取的是通过@Bean注解配置的bean

(2)如果根据名字还是匹配不到,解决方案为下面两个:

1)通过@primary设置某一个为主要的,比如:

@Component
@Primary
public class AutowiredTestService {

}

2)通过@Qualifier("autowiredtestservice2)告诉spring需要的那个bean的名字,比如:

public class AutowiredService {
    //首先会根据类型去spring容器中找(bytype) ,如果有多个类型, 会根据名字再去spring容器中找(byname)
    //容器中有两个AutowiredTestService类的bean,一个是通过@Component注解配置的bean(autowiredTestService)
    //还有一个是在SpringConfig配置文件中通过@Bean注解配置的bean(autowiredTestService2)
    //但是这里注入时的名字为autowiredTestService3,那他就会报错,因为根据名字autowiredTestService3,
    //他不知道应该找通过@Component注解配置的bean还是通过@Bean注解配置的bean,
    //这时候就需要通过@Qualifier("autowiredtestservice2)告诉spring需要的那个bean的名字
    @Autowired
    @Qualifier("autowiredTestService2") 
    AutowiredTestService autowiredTestService3; 
    // 虽然通过注入时的名字:autowiredTestService3找不到,
    // 但是可以通过@Qualifier中的autowiredTestService2找到
    @Override
    public String toString() {
        return "AutowiredService{" +
        "autowiredTestService=" + autowiredTestService3 +
        '}';
    }
}

3)@Autowired注解有一个required属性,当指定required属性为false时,意味着在容器中找相应类型的bean,如果找不到则忽略,而不报错(这一条是@Inject 和 @Resource 两个注解所没有的功能)。如果去容器中一个都没找到就会报错,通过@autowired(required=false)设置required=false,就不会报错了,比如:

@SpringBootTest  //需要加@SpringBootTest注解,否则这里需要自己去获取ioc容器才能从ioc容器中拿bean
public class AutowiredTest {

    @Autowired(required = false)  //如果去容器中一个都没找到,设置required=false,就不会报错了
    AutowiredService autowiredService3;

    public void testConstructor() {
        System.out.println(autowiredService3);
    }
}

(3)可以写在方法、构造函数、字段、参数上

上图中涉及的代码,先看这个代码,后面作为示例会有改动

@Service
public class ProductService {
}


@Service
public class StockService {
}


@Component
public class AutowiredService {
    private ProductService productService;
    private StockService stockService;

    //当这个构造函数被调用时,ProductService没有加@Autowired注解也会被自动的依赖注入进来
    public AutowiredService(ProductService productService){
        System.out.println(productService);
        this.productService = productService;
    }

}


@SpringBootTest  //需要加@SpringBootTest注解,否则这里需要自己去获取ioc容器才能从ioc容器中拿bean
public class AutowiredTest {

    @Autowired
    AutowiredService autowiredService3;

    @Test
    public void testConstructor() {
        System.out.println(autowiredService3);
    }
}

1)构造函数:

如果bean只有一个有参构造函数,就算省略@Autowired,也会自动注入构造函数的参数,比如:

public class AutowiredService {

    private ProductService productService;

    private StockService stockService;

    //当这个构造函数被调用时,ProductService没有加@Autowired注解也会被自动的依赖注入进来
    public AutowiredService(ProductService productService){
        System.out.println(productService);
        this.productService = productService;
    }

}

但是如果有多个有参构造函数,并且没有无参构造函数,就会报错,解决方式为使用@Autowired指定某一个构造函数,就是在要指定的那个构造函数上加@Autowired注解,比如:

public class AutowiredService {

    private ProductService productService;

    private StockService stockService;

    @Autowired  //在要指定的构造函数上加@Autowired注解,这样就不会报错,因为下面还有一个有参构造函数
    public AutowiredService(ProductService productService){
        System.out.println(productService);
        this.productService = productService;
    }

    public AutowiredService(ProductService productService, StockService stockService){
        this.productService = productService;
        this.stockService = stockService;
    }

}

但是这种构造函数上指定的@Autowired(required=false)会失效

2)参数:

如果想设置构造函数里面的参数为不是必须注入:可以单独去设置,就是在参数那里加@Autowired(required=false),比如:

@Autowired
public AutowiredService(@Autowired(required = false) ProductService productService){
    System.out.println(productService);
    this.productService = productService;
}

还可以单独的写在单元测试的方法上

@SpringBootTest  //需要加@SpringBootTest注解,否则这里需要自己去获取ioc容器才能从ioc容器中拿bean
public class AutowiredTest {

    //    @Autowired(required = false)
    //    AutowiredService autowiredService3;
    //@Autowired单独的写在下面的单元测试方法上,这里就不用注入了

    @Test
    public void testConstructor(@Autowired AutowiredService autowiredService) {
        System.out.println(autowiredService);
    }
}

3)方法:

spring会自动调用你的@Autowired的方法并进行自动注入需要的参数

//添加了@Autowired注解后,当Spring实例化完AutowiredService对象后,
@Autowired  //就会调用这个方法ltStockService()来进行自动注入这个方法里面的参数StockService
public void ltStockService(StockService stockService){
this.stockService = stockService;
}

//这里用来获取上面调用ltStockService()方法来进行自动注入StockService,并被赋值给stockService后的stockService
public StockService getStockService() {
    return stockService;
}

2、@Value

注入直接值(基本类型,String、List等)

        自动装配注解@Autowired、@Inject和@Resource 主要的是注入我们通过@Component等注解配置的类的实例对象,比如通过@Component把Student类配置成了bean,那么在Spring容器中就会有student这样的一bean,然后就可以通过自动装配(@Autowired)来去注入一个Student类型的bean,Spring容器就会把这个Student类型的bean注入进来。

        而@Value主要是用于注入一些基本的数据类型,比如Stiring、Integer,还有一些复杂数据类型如Map、Lsit等,因为这些数据类型都是由JDK提供的,不可能说去JDK提供的String这个类上面加一个@Component注解,所以这些数据类型通常用@Value进行注入

(1)方式一:直接把值写在@Value中的属性里

上图中涉及的代码,先看这个代码,后面作为示例会有改动

//自定义类User
@Component  //要使用@Value的话,需要先把当前这个类配置成一个bean
public class User {
    @Value("张三") //直接把值"张三"写在@Value中的属性里
    private String name;
    @Value("24") //直接把值"24"写在@Value中的属性里
    private Integer age;
    private Map<String, Integer> score;
    private List<String> hobbies;

    @Override //重写了toString()方法
    public String toString() {
        return "User{" +
        "name='" + name + '\'' +
        ", age=" + age +
        ", score=" + score +
        ", hobbies=" + hobbies +
        '}';
    }
}


//测试类ValueTest
@SpringBootTest
public class ValueTest {

    @Test
    public void Test01(@Autowired User user){
        System.out.println(user);
    }

}

 运行结果:

(2)方式二:对外部属性(SpringBoot配置文件)文件的引用

像前面那样把值直接写死在@Value上非常不利于后期的更改和维护,所以可以把这些值都写在一个properties文件中,要获取外部的属性文件需要通过@PropertuSource注解指定文件路径,然后在@Value中引用它,比如:

//zhangsan.properties文件
zhangsan.name = zhangsan
zhangsan.age = 24

示例代码:

@Component  //要使用@Value的话,需要先把当前这个类配置成一个bean
@PropertySource("zhangsan.properties")  //要获取外部的属性文件需要通过@PropertuSource指定文件路径 2.@Value("${zhangsan.name}")
public class User {
    //@Value("张三")
    @Value("${zhangsan.name}")
    private String name;
    //@Value("24")
    @Value("${zhangsan.age}")
    private Integer age;
    private Map<String, Integer> score;

    //剩下代码和上面一样.....没变
    
}

//测试类ValueTest的代码也没变

        在SpringBoot中,有一个默认的properties文件:application.properties,这个文件不需要单独的通过@PropertuSource指定文件路径

        并且在SpringBoot中,如果@Value("${}")的 {} 里面的属性值不存在,就会报错,因为SpringBoot更严格。在Spring中,如果@Value("${}")的 {} 里面的属性值不存在,会将属性名字注入到值中

        默认值机制:例如@Value("${zh.name}"),里面的zh.name不存在的话可以通过冒号设置默认值,更改后为@Value("${zh.name:MoRenZhi}"),zh.name如果不存在就会把 "MoRenZhi"注入到值中

        总结:如果是非SpringBoot配置文件,需要额外通过@PropertySource去指定属性文件的类路径,如果是SpringBoot配置文件(application.properties),就无需额外配置@PropertySource

(3)SPEL表达式(Spring Expression Language):@Value("#{}")

用来注入复杂数据类型Map、Lsit等,示例:

@Component  //要使用@Value的话,需要先把当前这个类配置成一个bean
@PropertySource("zhangsan.properties")  //1.要获取外部的属性文件需要通过@PropertuSource指定类路径 2.@Value("${zhangsan.name}")
public class User {
    //@Value("张三")
    @Value("${zhangsan.name}")
    private String name;
    //@Value("24")
    @Value("${zhangsan.age}")
    private Integer age;
    @Value("#{{'语文':'90','数学':'100'}}")
    private Map<String, Integer> score;
    @Value("#{{'唱歌,打球,写代码'}}")
    private List<String> hobbies;
    
    //剩下代码和上面一样.....
    
}

 

 

推荐: 

【Spring】使用@Bean和@Import注解配置Bean,与Bean的实例化_import和bean-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/m0_65277261/article/details/137257177?spm=1001.2014.3001.5501

【Spring】分别基于XML、注解和配置类实现Spring的IOC(控制反转)-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/m0_65277261/article/details/137237805?spm=1001.2014.3001.5501

【java多线程】线程池 ThreadPoolExecutor类和Executors工厂类以及线程池的最优大小-CSDN博客icon-default.png?t=N7T8https://blog.csdn.net/m0_65277261/article/details/137673188?spm=1001.2014.3001.5501

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

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

相关文章

冯喜运:4.16中东对抗风暴下,黄金原油市场何去何从?

黄金行情走势分析&#xff1a;4小时图布林道开始收口&#xff0c;昨日下探至下轨附近&#xff0c;也是此前的起涨低点2320启稳上升&#xff0c;十字K线配合单阳拉起&#xff0c;重新去摸高上轨。目前4小时图处于摸高当中。周线和日线留意多空转换&#xff0c;摸高之后是强势延续…

使用Scrapy选择器提取豆瓣电影信息,并用正则表达式从介绍详情中获取指定信息

本文同步更新于博主个人博客&#xff1a;blog.buzzchat.top 一、Scrapy框架 1. 介绍 在当今数字化的时代&#xff0c;数据是一种宝贵的资源&#xff0c;而网络爬虫&#xff08;Web Scraping&#xff09;则是获取网络数据的重要工具之一。而在 Python 生态系统中&#xff0c;S…

03-echarts如何画立体柱状图

echarts如何画立体柱状图 一、创建盒子1、创建盒子2、初始化盒子&#xff08;先绘制一个基本的二维柱状图的样式&#xff09;1、创建一个初始化图表的方法2、在mounted中调用这个方法3、在方法中写options和绘制图形 二、画图前知识1、坐标2、柱状图图解分析 三、构建方法1、创…

第七周学习笔记DAY.1-封装

学完本次课程后&#xff0c;你能够&#xff1a; 理解封装的作用 会使用封装 会使用Java中的包组织类 掌握访问修饰符&#xff0c;理解访问权限 没有封装的话属性访问随意&#xff0c;赋值也可能不合理&#xff0c;为了解决这些代码设计缺陷&#xff0c;可以使用封装。 面向…

RabbitMQ入门实战

文章目录 RabbitMQ入门实战基本概念安装快速入门单向发送多消费者 RabbitMQ入门实战 官方&#xff1a;https://www.rabbitmq.com 基本概念 AMQP协议&#xff1a;https://www.rabbitmq.com/tutorials/amqp-concepts.html 定义&#xff1a;高级信息队列协议&#xff08;Advanc…

LangChain入门:19.探索结构化工具对话

引言 在人工智能的浪潮中&#xff0c;对话代理技术正逐渐成为企业和开发者关注的焦点。LangChain&#xff0c;作为对话代理领域的一颗新星&#xff0c;自2021年9月诞生以来&#xff0c;以其强大的功能和灵活的应用场景迅速赢得了市场的认可。本文将带你深入了解LangChain中的S…

【从浅学到熟知Linux】进程控制上篇=>进程创建、进程终止与进程等待(含_exit与exit的区别、fork函数详解、wait与waitpid详解)

&#x1f3e0;关于专栏&#xff1a;Linux的浅学到熟知专栏用于记录Linux系统编程、网络编程等内容。 &#x1f3af;每天努力一点点&#xff0c;技术变化看得见 文章目录 进程创建fork函数写时拷贝 进程退出进程退出操作系统做了什么&#xff1f;进程退出场景进程退出的常见方法…

openstack修改实例名称但是gnocchi监控数据中实例名称没有变更的问题处理

文章目录 一、问题描述二、调研过程1、变更实例名称2、查看grafana中的监控数据3、libvirt服务中的xml文件4、现有的监控数据流转架构 总结 一、问题描述 openstack修改实例名称但是gnocchi监控数据中实例名称没有变更的问题处理。 通过修改实例名称的功能修改了实例名称&…

大模型赋能:爬虫技术的全新革命

大模型加持下的爬虫技术革新&#xff1a;从BS4到提示工程的飞跃 在爬虫技术的演进历程中&#xff0c;内容解析一直是一个核心环节。传统的爬虫技术&#xff0c;如使用BeautifulSoup&#xff08;BS4&#xff09;等工具&#xff0c;需要逐个解析网页内容&#xff0c;通过XPath或C…

XILINX 7系列时钟资源

文章目录 前言一、时钟概要1.1、CC1.2、BUFR、BUFIO、BUFMR1.3、CMT1.4、BUFH1.5、BUFG 二、时钟路由资源三、CMT 前言 本文主要参考xilinx手册ug472 一、时钟概要 7系列FPGA时钟资源主要有CC、BUFR、BUFIO、BUFMR、CMT、BUFG、BUFH和GTE_COMMON 1.1、CC “CC”&#xff0…

OpenHarmony开发案例:【自定义通知】

介绍 本示例使用[ohos.notificationManager] 等接口&#xff0c;展示了如何初始化不同类型通知的通知内容以及通知的发布、取消及桌面角标的设置&#xff0c;通知类型包括基本类型、长文本类型、多行文本类型、图片类型、带按钮的通知、点击可跳转到应用的通知。 效果预览&am…

TensorFlow实战Google深度学习框架 PDF书籍分享

今天又来给大家推荐一本TensorFlow方面的书籍<TensorFlow实战Google深度学习框架>。本书适用于想要使用深度学习或TensorFlow的数据科学家、工程师&#xff0c;希望了解大数据平台工程师&#xff0c;对人工智能、深度学习感兴趣的计算机相关从业人员及在校学生等。 下载当…

【数据结构与算法】用两个栈实现一个队列

题目 用两个栈&#xff0c;实现一个队列功能 add delete length 队列 用数组可以实现队列&#xff0c;数组和队列的区别是&#xff1a;队列是逻辑结构是一个抽象模型&#xff0c;简单地可以用数组、链表实现&#xff0c;所以数组和链表是一个物理结构&#xff0c;队列是一个逻…

Cannot access ‘androidx.activity.FullyDrawnReporterOwner‘

Android Studio新建项目就报错&#xff1a; Cannot access ‘androidx.activity.FullyDrawnReporterOwner’ which is a supertype of ‘cn.dazhou.osddemo.MainActivity’. Check your module classpath for missing or conflicting dependencies 整个类都报错了。本来原来一直…

阿里面试:DDD中的实体、值对象有什么区别?

在领域驱动设计&#xff08;DDD&#xff09;中&#xff0c;有两个基础概念&#xff1a;实体&#xff08;Entity&#xff09;和值对象&#xff08;Value Object&#xff09;。 使用这些概念&#xff0c;我们可以把复杂的业务需求映射成简单、明确的数据模型。正确使用实体和值对…

【环境】原则

系列文章目录 【引论一】项目管理的意义 【引论二】项目管理的逻辑 【环境】概述 【环境】原则 一、培养项目系统性思维 1.1 系统性思维 1.2 系统性思维的价值 1.3 建模和推演&数字孪生 二、项目的复杂性和如何驾驭复杂性 2.1 复杂性的三个维度 2.2 如何驾驭复杂性 三、…

【御控物联】Java JSON结构转换(4):对象To对象——规则属性重组

文章目录 一、JSON结构转换是什么&#xff1f;二、术语解释三、案例之《JSON对象 To JSON对象》四、代码实现五、在线转换工具六、技术资料 一、JSON结构转换是什么&#xff1f; JSON结构转换指的是将一个JSON对象或JSON数组按照一定规则进行重组、筛选、映射或转换&#xff0…

谷歌pixel6/7pro等手机WiFi不能上网,显示网络连接受限

近期在项目中遇到一个机型出现的问题,先对项目代码进行排查,发现别的设备都能正常运行,就开始来排查机型的问题,特意写出来方便后续查看,也方便其它开发者来自查。 设备机型:Pixel 6a 设备安卓版本:13 该方法无需root,只需要电脑设备安装adb(即Android Debug Bridge…

GPT提示词分享 —— 解梦

&#x1f449; 对你描述的梦境进行解读。 我希望你能充当一个解梦者。我将给你描述我的梦&#xff0c;而你将根据梦中出现的符号和主题提供解释。不要提供关于梦者的个人意见或假设。只提供基于所给信息的事实性解释。 GPT3.5的回答 GPT3.5 &#x1f447; 感觉有点傻&#xf…

申请免费https证书

https证书是什么&#xff1a; https证书是指在http超文本传输协议的前提下安装部署了SSL/TLS证书后形成的全新协议&#xff0c;https安全超文本传输协议。在https证书部署完成后&#xff0c;服务器端和浏览器端进行的信息交互的过程中会有加密层保护&#xff0c;使得原本明文传…