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

目录

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,名字是autowiredTestServiceAutowiredTestService autowiredTestService;@Overridepublic String toString() {return "AutowiredService{" +"autowiredTestService=" + autowiredTestService +'}';}
}@Configuration
public class SpringConfig {@Bean  //这里配置bean的类型是AutowiredTestService,名字是autowiredTestService2public 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;@Testpublic 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,名字是autowiredTestService2AutowiredTestService autowiredTestService2;@Overridepublic 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找到@Overridepublic 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 {@AutowiredAutowiredService autowiredService3;@Testpublic 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单独的写在下面的单元测试方法上,这里就不用注入了@Testpublic 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 {@Testpublic 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.hqwc.cn/news/620929.html

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

相关文章

基于注解以及配置类使用SpringIoc

四 基于注解方式使用SpringIoc 和 XML 配置文件一样&#xff0c;注解本身并不能执行&#xff0c;注解本身仅仅只是做一个标记&#xff0c;具体的功能是框架检测到注解标记的位置&#xff0c;然后针对这个位置按照注解标记的功能来执行具体操作。 本质上&#xff1a;所有一切的…

extends继承

目录 什么时候用继承? 继承的格式? 继承的特点 子类可以继承父类的哪些呢&#xff1f; 是否可以继承父类的构造方法呢&#xff1f; 是否可以继承成员变量&#xff1f; 是否可以继承成员方法&#xff1f; 在Java中&#xff0c;extends关键字用于实现继承关系。通过使用…

Go gin框架(详细版)

目录 0. 为什么会有Go 1. 环境搭建 2. 单-请求&&返回-样例 3. RESTful API 3.1 首先什么是RESTful API 3.2 Gin框架支持RESTful API的开发 4. 返回前端代码 go.main index.html 5. 添加静态文件 main.go 改动的地方 index.html 改动的地方 style.css 改动…

洛谷 P9532 [YsOI2023] 前缀和

题目背景 Ysuperman 模板测试的试机题。 小心立秋&#xff0c;小心秋丽。 题目描述 立秋有一个长度为 n 的数组 a&#xff0c;所有数字都是正整数&#xff0c;并且除了其中第一个数字以外其它数字都等于前面所有数字的和。 例如&#xff0c;数组 [1,1,2,4,8,16] 就有可能是…

基于U-Net的图像分割算法介绍

U-Net是一种用于图像分割的深度学习架构,其设计初衷是用于生物医学图像分割,尤其是医学影像中的细胞分割任务。U-Net结构独特,具有编码器-解码器结构,能够有效地捕捉图像中的局部和全局信息,并在像素级别上进行精确的分割。 相关论文: U-Net: Convolutional Networks for…

RISCV指令集体系简读之RV32I

RV32I 指令格式 用于寄存器-寄存器操作的R类型指令用于短立即数和访存load操作的I型指令用于访存store操作的s型指令用于条件跳转操作的B类型指令用于长立即数的U型指令用于无条件跳转的J型指令 特点&#xff1a; 所有指令都是32bits&#xff0c; 简化了指令解码&#xff1b;…

Python数据可视化库—Bokeh与Altair指南【第161篇—数据可视化】

&#x1f47d;发现宝藏 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。【点击进入巨牛的人工智能学习网站】。 在数据科学和数据分析领域&#xff0c;数据可视化是一种强大的工具&#xff0c;可以帮助我们…

Mac电脑安装蚁剑

1&#xff1a; github 下载源码和加载器&#xff1a;https://github.com/AntSwordProjectAntSwordProject GitHubAntSwordProject has 12 repositories available. Follow their code on GitHub.https://github.com/AntSwordProject 以该图为主页面&#xff1a;antSword为源码…

关于SpringCloud,你了解多少?

Why SpringCloud&#xff1f; Spring cloud 是一系列框架的有序集合。它利用 spring boot 的开发便利性巧妙地简化了分布式系统基础设施的开发&#xff0c;如服务发现注册、配置中心、消息总线、负载均衡、断路器、数据监控等&#xff0c;都可以用 spring boot 的开发风格做到一…

MercadoLibre(美客多)入仓预约系统操作流程-自动化约号(开篇)

目录 一、添加货件信息 二、输入货件信息 三、选择发货 四、填写交货日期 五、注意事项 MercadoLibre&#xff08;美客多&#xff09;于2021年10月18号上线了新预约入仓系统&#xff0c;在MercadoLibre美客多平台上&#xff0c;新入仓预约系统是一项非常重要的功能&#x…

通用设计的四大原则,大厂设计师带案例讲解!

作为数字产品设计师&#xff0c;在进行产品设计时要考虑产品的各种因素&#xff0c;例如功能、美观、安全等&#xff0c;要尽可能地满足所有用户的需求&#xff0c;做出对所有用户都尽可能公平的解决方案。但是&#xff0c;对于新手来说&#xff0c;在实际进行产品设计时&#…