Spring之@Value注解

news2025/2/24 11:57:18

前言

@Value注解在Spring的依赖注入中占据重要地位,这里对@Value注解的作用进行演示以及扩展

作用

  • 注入字符串
  • 注入属性
  • 注入bean
  • 其他

代码准备

创建两个普通的bean

@Component
public class ValueComponent {
}
@Component
public class Foo {

    private String sign;

    public Foo() {
        this.sign = UUID.randomUUID().toString().replaceAll("-", "");
    }

    public String getSign() {
        return sign;
    }

    public void setSign(String sign) {
        this.sign = sign;
    }
}

创建配置文件val.properties

key=source
source=spring
color=blank,white,red

创建配置类

@ComponentScan("com.test.val")
@PropertySource("classpath:val.properties")
public class AppConfig {

}

创建启动类

public class Main {

    public static void main(String[] args) {

        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class);
    }
}

示例

注入字符串

@Component
public class ValueComponent {

    @Value("hello world")
    private String helloWorld;

}

注入属性

注入普通属性
@Component
public class ValueComponent {

    @Value("${key}")
    private String key;

}

注入嵌套属性
@Component
public class ValueComponent {

    @Value("${${key}}")
    private String nestKey;

}

 注入的属性不存在
@Component
public class ValueComponent {

    @Value("${server.port}")
    private String absentKey;

}

PS : Spring默认情况下使用的是宽松模式, 解析不了的属性等于注入了字符串

注入的属性不存在,使用默认值
@Component
public class ValueComponent {

    @Value("${server.port:8080}")
    private String absentDefaultKey;

}

注入bean及其属性

@Component
public class ValueComponent {

    @Value("#{foo}")
    private Foo foo;

    @Value("#{foo['sign']}")
    private String sign;
}

其他

@Component
public class ValueComponent {

    @Value("https://www.baidu.com/")
    private URL url;

    @Value("classpath:val.properties")
    private Resource resource;

}

属性注入优先级问题

创建配置文件val2.properties 

key=source2
source=spring2

 修改配置文件

@ComponentScan("com.test.val")
@PropertySources({@PropertySource("classpath:val.properties"), @PropertySource("classpath:val2.properties")})
public class AppConfig {

}

注入普通属性key

@Component
public class ValueComponent {

    @Value("${key}")
    private String key;

}

Spring默认情况下创建的Environment是StandardEnvironment,会添加两个默认PropertySource : systemProperties systemEnvironment

系统默认添加的两个PropertySource优先级最高,使用@PropertySource(@PropertySources)注解导入的propertySource,越先解析优先级越低

当前环境的PropertySource排序

systemProperties > systemEnvironment > val2.properties > val1.properties 

如果在优先级较高的PropertySource里面找到了相关属性,则直接返回不会查找优先级较低的PropertySource了

Springboot对Spring做了很多扩展, 存在很多PropertySource

对@Value属性注入的扩展

如果beanFactory中不存在embeddedValueResolvers则会添加一个默认的embeddedValueResolvers

AbstractApplicationContext#finishBeanFactoryInitialization

DefaultListableBeanFactory#doResolveDependency

AbstractBeanFactory#resolveEmbeddedValue

在上述的前提下我们可以自定义一个StringValueResolver来解析@Value注解传入的字符串

创建MergedResolver对象

public class MergedResolver implements StringValueResolver {

    private PropertySources propertySources;

    private final PropertySourcesPropertyResolver defaultResolver;

    private final PropertySourcesPropertyResolver resolver1;

    private final PropertySourcesPropertyResolver resolver2;

    public MergedResolver(PropertySources propertySources) {
        this.propertySources = propertySources;

        defaultResolver = new PropertySourcesPropertyResolver(this.propertySources);

        resolver1 = new PropertySourcesPropertyResolver(this.propertySources);
        resolver1.setPlaceholderPrefix("$[");
        resolver1.setPlaceholderSuffix("]");

        resolver2 = new PropertySourcesPropertyResolver(this.propertySources);
        resolver2.setPlaceholderPrefix("$(");
        resolver2.setPlaceholderSuffix(")");
    }

    @Override
    public String resolveStringValue(String strVal) {
        if (strVal.startsWith("$[")) {
            return resolver1.resolvePlaceholders(strVal);
        } else if (strVal.startsWith("$(")) {
            return resolver2.resolvePlaceholders(strVal);
        } else {
            return defaultResolver.resolvePlaceholders(strVal);
        }
    }
}

创建StringValueResolverImporter对象

public class StringValueResolverImporter implements ImportBeanDefinitionRegistrar, EnvironmentAware {

    private Environment environment;

    @Override
    public void setEnvironment(Environment environment) {
        this.environment = environment;
    }

    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        DefaultListableBeanFactory beanFactory = (DefaultListableBeanFactory) registry;
        // 添加自定义EmbeddedValueResolver
        // 自定义的EmbeddedValueResolver要兼容默认的EmbeddedValueResolver,否则默认的@Value功能全部失效
        // 一定要兼容默认的EmbeddedValueResolver 一定要兼容默认的EmbeddedValueResolver 一定要兼容默认的EmbeddedValueResolver
        // 重要的事情说三遍 ! ! !
        beanFactory.addEmbeddedValueResolver(new MergedResolver(((StandardEnvironment) environment).getPropertySources()));
    }

}

修改配置文件

@ComponentScan("com.test.val")
@Import(StringValueResolverImporter.class)
@PropertySources({@PropertySource("classpath:val.properties"), @PropertySource("classpath:val2.properties")})
public class AppConfig {

}

修改ValueComponent

@Component
public class ValueComponent {

    @Value("${key}")
    private String key1;

    @Value("$[key]")
    private String key2;

    @Value("$(key)")
    private String key3;

}

运行Main方法,查看运行结果

Springboot对@Value类型转换的扩展

修改ValueComponent

@Component
public class ValueComponent {

    @Value("${color}")
    private List<String> color;
}

如果使用的是SpringBoot,会将字符串以逗号分割,然后放入list中

主要原因是Springboot给BeanFactory添加了一个ApplicationConversionService,这个类的默认构造方法会添加很多convert

通过源码我们知道了这个扩展点可以使用@Delimiter指定分隔符,然后默认分隔符是逗号

使用Spring达到同样效果

复用StringValueResolverImporter代码

 修改val.properties

key=source
source=spring
color=blank,white,red
car=redCar;whiteCar;blackCar

修改ValueComponent

@Component
public class ValueComponent {

    @Value("${color}")
    private List<String> color;

    @Value("${car}")
    @Delimiter(";")
    private List<String> car;
}

运行Main方法,查看运行结果

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

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

相关文章

算法——位运算(一篇搞定)

本专栏为大家分享本人学习算法遇到的不同类型的题目以及解析! 此篇文章给大家分享一些关于位运算算法的题目,在开篇还讲述了常见位运算的公式以及题目,如果对您有帮助,麻烦点个关注,如有错误,请您指出! 1.常见位运算总结(包含5道题目) 1.1基础位运算 运算符操作<<左移&…

Google XSS Game Level 6 通关方式

文章目录 链接&#xff1a;[Google XSS Game](#https://xss-game.appspot.com/)Level 6 - Follow the &#x1f407;思路1 &#xff08;当然&#xff0c;我使用这个方式没有成功&#xff0c;所以才来记录下&#xff09;解法2 【最简单的解法】需要注意的一个小问题 链接&#x…

卡特兰数的部分解析(1)

直接开始 关于卡特兰数有三个公式 这里会解析一下第二个公式的含义 直接上公式 C n C 2 n n − C 2 n n − 1 或者是 C n C 2 n n − C 2 n n 1 C_n C_{2n}^n - C_{2n}^{n-1} 或者是 C_n C_{2n}^n - C_{2n}^{n1} Cn​C2nn​−C2nn−1​或者是Cn​C2nn​−C2nn1​ 解析…

网络编程—DAY5

select实现的TCP并发服务器 #include <myhead.h> #define SER_PORT 8888 #define SER_IP "192.168.117.96"int main(int argc, const char *argv[]) {int sfd -1;sfd socket(AF_INET,SOCK_STREAM,0);if(sfd -1){perror("socket");return -1;}prin…

Python文件操作相关知识点(读取/写入数据)

1.open函数的定义 &#xff08;1&#xff09;open函数的简要概述 open&#xff08;filename&#xff09;函数接受一个参数——要打开文件的名称&#xff0c;Python在当前执行的文件所在的目录中查找指定的文件。并返回一个表示文件的对象。 open&#xff08;&#xff09;函数…

基于python智慧社区家政服务系统的设计与实现flask-django-nodejs-php

随着现代网络技术发展&#xff0c;对于智慧社区家政服务系统的设计现在正处于发展的阶段&#xff0c;所以对的要求也是比较严格的&#xff0c;要从系统的功能和用户实际需求来进行对系统制定开发的发展方式&#xff0c;依靠网络技术的的快速发展和现代通讯技术的结合为人们带来…

MyBatis是纸老虎吗?(四)

在《MyBatis是纸老虎吗&#xff1f;&#xff08;三&#xff09;》这篇文章中我们一起梳理了MyBatis配置文件的解析流程&#xff0c;并详细介绍了其中的一些常见节点的解析步骤。通过梳理&#xff0c;我们弄清楚了MyBatis配置文件中的一些常用配置项与Java Bean之间的对应关系&a…

threejs之贴图原理

// 导入threejs import * as THREE from "three"; // 导入轨道控制器 import { OrbitControls } from "three/examples/jsm/controls/OrbitControls.js";// 创建场景 const scene new THREE.Scene();// 创建相机 const camera new THREE.PerspectiveCame…

使用Java JDBC连接数据库

在Java应用程序中&#xff0c;与数据库交互是一个常见的任务。Java数据库连接&#xff08;JDBC&#xff09;是一种用于在Java应用程序和数据库之间建立连接并执行SQL查询的标准API。通过JDBC&#xff0c;您可以轻松地执行各种数据库操作&#xff0c;如插入、更新、删除和查询数…

有没搞错!花了大价钱的激光孔设计性能竟然不如普通通孔?

高速先生成员--黄刚 老话说得好&#xff0c;一分耕耘一分收获&#xff0c;又或者另外一句&#xff0c;有什么付出就会得到多少收获。我们都不会去怀疑这些话的正确性。但是把这两句话用到PCB领域中&#xff0c;用了好的加工工艺后&#xff0c;PCB板的性能就一定会比用普通工艺要…

视频素材免费下载素材库哪里有?推荐8个高清无水印素材网

在这个数字化时代&#xff0c;无论是专业的内容创作者还是日常的社交媒体使用者&#xff0c;我们都会寻找高质量的素材来丰富我们的作品或帖子。从令人震撼的摄影作品到高分辨率的视频素材&#xff0c;再到生动的GIF和必需的设计元素&#xff0c;素材的需求无处不在。 视频素材…

XXL-JOB完全开发手册(一篇学会XXL-JOB所有知识点)

目录 1、什么是XXL-JOB 1.1、XXL-JOB简介 1.2、XXL-JOB构成 调度模块&#xff08;调度中心&#xff09;&#xff1a; 执行模块&#xff08;执行器&#xff09;&#xff1a; 任务&#xff1a; 1.3、XXL-JOB总结 ​编辑 2、XXL-JOB原理 2.1、执行器的注册和发现 2.2、调度中心调用…

基于华为ensp的企业网络规划(新版)

第一章 项目概述 1.1 项目总体描述 假设某大型公司总部在北京、在重庆设置分部&#xff0c;总部和分部均有研发部、市场部、财务部等部门&#xff0c;现在要求进行网络规划与设计&#xff0c;实现分部和总部能够进行网络连通。为了保证数据安全&#xff0c;在总部和分部之间可…

教师怎么发成绩才不会被投诉

在当今社会&#xff0c;教育的重要性日益凸显&#xff0c;而学生的成绩作为衡量教育效果的重要指标之一&#xff0c;备受家长和学生的关注。然而&#xff0c;教师在发布成绩时&#xff0c;稍有不慎就可能引发家长的不满和投诉。那么&#xff0c;教师该如何发成绩才能避免被投诉…

【Spring Cloud Gateway】路由配置uri三种方式及区别

websocket配置方式 ws:// 或 wss:// 开头的 URI&#xff0c;表示配置的是支持 Websocket 协议的目标地址。 这种方式适用于需要与客户端建立长连接、实现双向通信的场景&#xff0c;比如实时消息推送、即时聊天等。 使用 Websocket 配置方式可以让 Spring Cloud Gateway 能够…

【物联网】Modbus 协议及应用

Modbus 协议简介 QingHub设计器在设计物联网数据采集时不可避免的需要针对Modbus协议的设备做相关数据采集&#xff0c;这里就我们的实际项目经验分享Modbus协议 简介 Modbus由MODICON公司于1979年开发&#xff0c;是一种工业现场总线协议标准。1996年施耐德公司推出基于以太…

虚拟机VMware上 centos7 的网络配置

第一步&#xff1a;权限的切换 由普通用户切换到管理者/超级用户 用户名为&#xff1a;root 密码为&#xff1a;自己安装 linux 时第一次设置的密码 su -root管理者/超级用户的命令提示符是“#”&#xff0c;普通用户的命令提示符是“$”。当看到你的命令提示符为“$”时&…

《世界之外》玩家闹上315,乙游打响维权大战

315维权微博的评论区&#xff0c;竟然被举报网易的玩家占领了。 玩家举报网易乙游《世界之外》虚假宣传侵害消费者权益&#xff0c;在游戏中设置排行榜和专属商店将玩家分为三六九等&#xff0c;诱导玩家消费氪金&#xff0c;强烈要求网易打开退款通道。 目前大批玩家举报的举…

如何在三个简单步骤中为对象检测标注图像

初始通过彻底清洗和处理原始图像数据来奠定有效对象检测注释的基础。选择适合的工具、方法和清晰的注释过程指南来建立注释工作空间。通过在图像中划定对象并附上类别标签来执行注释&#xff0c;随后进行细致的核验&#xff0c;以确保数据集的精确性和完整性。 图像注释是计算…

klipper源码分析之simulavr测试

分析Klipper源码&#xff0c;有时需要结合下位机一起分析&#xff0c;这样才能更加全面的了解Klipper的工作原理。如果手头上有打印机主板&#xff0c;电脑当做上位机运行Klipper&#xff0c;这样是比较方便。如果手头上没有打印机主板&#xff0c;可以用simulavr模拟AVR下位机…