所有分类
  • 所有分类
  • 未分类

Spring注解-@Value的使用

简介

说明

本文用示例介绍SpringBoot的@Value的用法。

等效的方法

也可以通过如下方法获得配置的值:

1.@ConfigurationProperties

2. ApplicationContextHolder.getContext().getEnvironment().getRequiredProperty(key);
    ApplicationContextHolder见:SpringBoot-静态获得Bean的工具类 – 自学精灵

分类

@Value属性注入功能根据注入的内容来源可分为两类:通过配置文件的属性注入和通过非配置文件的属性注入。 

通过配置文件的注入根据配置文件的来源又可分为两类:

  1. 默认的Spring Boot会自动加载的配置文件application.properties中的属性;
  2. 自定义配置文件中的属性,需要先通过@PropertySource加载。

非配置文件注入的类型又分为:

  • 注入普通字符串
  • 注入操作系统属性
  • 注入表达式结果
  • 注入其他Bean属性
  • 注入文件资源
  • 注入URL资源

@ConfigurationProperties与@Value

@ConfigurationProperties@Value
类型Map、内部类、对象等。不支持内部类、对象。
spEl表达式不支持支持
JSR303数据校验支持不支持
功能一个列属性批量注入单属性注入

格式问题

特殊字符

key和value都不能包含的字符:_、-、~、!、@、#、$、*、+、:等

如果value里有这些字符,用双引号包裹起来即可。

宽松绑定

以下方式写法相当于同一种写法。比如:yml用helloname,代码可以用helloName获取其值。

所有位置都支持宽松绑定,包括prefix、value。

helloname
helloName
HELLONAME
hello-name
hello_name
HELLO-NAME

基于配置文件注入

常用操作

application.properties

user.name=admin

@Value在Bean中的使用(注解到字段上)

@RestController
public class ValueController {
    @Value("${user.name}")
    private String name;

    @Value("${user.password}")
    private String password;
}

 @Value在Bean中的使用(注解到参数上)

@RestController
public class ValueController {
    private String password;

    public ValueController(@Value("${user.password}")private String password){
        this.password = password
    }
}

还可以注入数组和列表形式

application.properties

tools=car,train,airplane
// 注入数组(自动根据","分割)
@Value("${tools}")
private String[] toolArray;

// 注入列表形式(自动根据","分割)
@Value("${tools}")
private List<String> toolList;

可拼接字符串

application.yml

custom:
  name: hello

Controller

package com.knife.controller;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class HelloController {
    @Value("my name is: " + "${custom.name}")
    private String name;

    @GetMapping("/test")
    public String test() {

        return name;
    }
}

访问:http://localhost:8080/test

结果:

可拼接配置字符串

application.yml

app:
  prefix: Hello
  suffix: World

Controller

package com.knife.example.controller;

import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;

@Slf4j
@Api(tags = "测试")
@RestController
public class HelloController {

    @Value("${app.prefix} ${app.suffix}")
    private String combinedValue;

    @ApiOperation("测试1")
    @GetMapping("test")
    public String test1() {
        return combinedValue;
    }
}

测试

访问:http://localhost:8080/test

结果

可指定配置文件

 my.properties

user.password=pwd123

@Value在Bean中的使用(注解到字段上)

@PropertySource("classpath:my.properties")
@RestController
public class ValueController {
    @Value("${user.name}")
    private String name;

    @Value("${user.password}")
    private String password;
}

 @Value在Bean中的使用(注解到参数上)

@PropertySource("classpath:my.properties")
@RestController
public class ValueController {
    private String password;

    ValueController(@Value("${user.password}") String password){
        this.password = password
    }
}

默认值注入

概述

说明

使用${} 或 #{}进行属性注入时,当获取不到值时可以设置默认值。

示例

// 若未配置ip,则使用默认值(127.0.0.1)
@Value("${ip:127.0.0.1}")
private String ip;

// 若未配置custom.myProperty,则使用默认值(Hello)
@Value("${custom.myProperty:Hello}")
private String myProperty;

// 如果属性中未配置custom.myProperty,则使用默认值(空字符串)
@Value("${custom.myProperty:}")
private String myProperty;

// 若未配置custom.myProperty,则使用默认值(null)
@Value("${custom.myProperty::#{null}}")
private String myProperty;

// 如果属性中配置custom.myProperty为空,则使用默认值(Hello)
@Value("#{${custom.myProperty}?:'Hello'}")
private String myProperty;

// 如果系统属性中未获取到port的值,则使用默认值(8888)。
@Value("#{systemProperties['port']?:'8888'}")
private String port;

${} 和 #{}获取不到值的区别

对于“获取不到值”,${}和#{}不一样。

情景1:无key,无value(即:没有配置)

${}:不报错,程序中myProperty的值为“Hello” 

@Value("${custom.myProperty:Hello}")
private String myProperty;

#{}:启动时报错。

@Value("#{${custom.myProperty}?:'Tony'}")
private String myProperty;

报错信息:Caused by: java.lang.IllegalArgumentException: Could not resolve placeholder ‘custom.myProperty’ in value “#{${custom.myProperty}?:’Tony’}”

情景2:有key,无value(即:配置为空值)

application.yml

custom:
  myProperty:

 ${}:不报错。程序中myProperty的值为“” (空字符串)

@Value("${custom.myProperty:Tony}")
private String myProperty;

#{}:不报错。程序中myProperty的值为“Tony” 

@Value("#{${custom.myProperty}?:'Tony'}")
private String myProperty;

SpEL

说明

SpEL(Spring Expression Language)即Spring表达式语言,可以在运行时查询和操作数据。使用#{…}作为定界符, 所有在大括号中的字符都将被认为是 SpEL。

SpEL详见:Spring-SpEL的用法 – 自学精灵

示例

// 注入普通字符串,相当于直接给属性默认值
@Value("这是我的字符串")
private String wechatSubscription;

// 注入配置
@Value("#{config.tool}")
private String tool;

//  注入操作系统属性
@Value("#{systemProperties['os.name']}")
private String systemPropertiesName;

// 注入表达式结果
@Value("#{ T(java.lang.Math).random() * 100.0 }")
private double randomNumber;

// 注入列表形式(自动根据"|"分割)
@Value("#{'${words}'.split('\\|')}")
private List<String> numList;

// 注入文件资源
@Value("classpath:config.xml")
private Resource resourceFile;

// 注入URL资源
@Value("http://www.choupangxia.com")
private URL homePage;
0

评论0

请先

显示验证码
没有账号?注册  忘记密码?

社交账号快速登录