@Value注解位于springbeans中,以下是@Value注解的源码

package org.springframework.beans.factory.annotation;

import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target({ElementType.FIELD, ElementType.METHOD, ElementType.PARAMETER, ElementType.ANNOTATION_TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Value {
    String value();
}

由上可以看出:

  1. @Value可以修饰属性方法参数注释类型
  2. 编译器会将 @Value注解的信息保留在 .class 文件中,并且能被虚拟机读取
  3. @Value可以出现javadoc 中。
  4. 该注解中的String value(); 意味着,@Value能指定参数

@Value的用法

@Value可以获取配置文件中的值,设置属性,也可以引用Bean的属性值。下面通过SpringBoot项目讲解@Value的用法

一、@Value引用配置文件中的属性值

使用@Value引用配置文件中的属性值的方式

@Value(“${属性名}”)

application.yml的配置

application.yml文件的配置如下

ymlname: only-yml
student:
  name: yml里的name
  age: 20
  tel : 666

application.properties的配置

application.properties文件的配置如下
在这里插入图片描述

测试用Controller

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@Controller
public class TestValueController {

    // 只在application.yml中配置
    @Value("${ymlname}")
    private String ymlname;
    // 只在application.properties中配置
    @Value("${propname}")
    private String propname;

    // application.yml和application.properties均有该配置
    @Value("${student.name}")
    private String name;

    // 配置文件中的字段名和属性名不一致
    @Value("${student.age}")
    private int nianling;

    // application.yml和application.properties没有该配置,设置默认值
    @Value("${student.score:100}")
    private int score;

    // application.yml有该配置,同时设置默认值
    @Value("${student.tel:888}")
    public int tel;


    @ResponseBody
    @RequestMapping(value = "/test")
    public String testValue() {
        return "ymlname —— " + ymlname + "<br/>" +
                "propname —— " + propname + "<br/>" +
                "name —— " + name + "<br/>" +
                "nianling —— " + nianling + "<br/>" +
                "score —— " + score + "<br/>" +
                "tel —— " + tel;
    }
}

启动项目查看结果

启动SpringBoot项目浏览器输入localhost:8080/test界面显示如下
在这里插入图片描述

二、@Value作用于静态变量

正常情况下 @Value不可作用静态属性如下例。
启动类上做如下修改

import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
public class SpringbootTestApplication {

    @Value("${student.name}")
    public static String name;

    public static void main(String[] args) {
        SpringApplication.run(SpringbootTestApplication.class, args);

        System.out.println("name: " + name);
    }

}

打印结果如下
在这里插入图片描述
通过上例可以看出,使用@Value注解修饰静态属性,启动项目时不会报错,但是也不会给该静态属性设置值。
可以通过set方法静态属性设置配置文件中的属性值。

    public static String name;
    
    @Value("${student.name}")
    public void setName(String param) {
    	name = param;
    }

三、@Value引用Bean的属性值

使用@Value引用Bean的属性值的方式和引用配置文件中的属性值方式类似。使用方式

@Value(“#{bean名字.属性值}”)

以通过@Value注解引用User实例的name属性值为例
User类:

import lombok.AllArgsConstructor;
import lombok.Data;

@Data
@AllArgsConstructor
public class User {

    private String name;

    private String password;
}

配置一个TestConfig类,用于产生一个name为zhangsan,password为66666的名为userbean实例交由spring容器管理

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class TestConfig {

    @Bean(name = "user")
    public User getUser() {
        return new User("zhangsan","66666");
    }
}

TestBeanPro 类用于测试,其有一个userName属性,通过@Value注解将容器中名为user的bean的name属性注入userName。@PostConstruct注解的方法该类构造方法执行完成后执行。在本例中,该初始化方法用于打印user的name属性是否引用成功

import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;

@Slf4j
@Component
public class TestBeanPro {

    @Value("#{user.name}")
    private String userName;

    @PostConstruct
    public void init() {
        log.info("***************************** userName:{}.", userName);
    }
}

启动项目后通过控制台日志可以看到userName的值为zhangsan。控制台日志如下
在这里插入图片描述

总结

从以上测试结果可以看出:

  1. application.yml和application.properties中配置的值都可以通过@Value注解获取
  2. 若application.yml和application.properties同时配有同一个变量的值,则以application.yml的值为主;
  3. 配置文件中的字段名和@Value修饰的属性名可以不一致
  4. @Value若从配置文件中获取不到值,则设置的默认值生效
  5. 若配置文件中有配置,则默认值生效

通过对@Value的以上分析我们还不难看出,SpringBoot加载配置文件的顺序为.yml > .properties。即yml类型优先级高于properties类型的配置文件。



使用@Value给静态变量注入

最近做项目的时候,给static变量赋值使用 @value注解 ,结果 获取一直为null

1spring不允许/支持把值注入静态变量2Spring@Value依赖注入依赖set方法
 
3set方法是普通的对象方法
 
4static变量是类的属性,static没有set方法

SpringBoot中使用@Value()只能给普通变量注入值,不能直接静态变量赋值

例如,application-dev.properties配置文件有如下配置:

img

给普通变量赋值时,直接在变量声明之上添加@Value()注解即可,如下所示

img

当要给静态变量注入值的时候,若是在静态变量声明之上直接添加@Value()注解是无效的,例如

img

虽然没有编译运行上的报错,经调试可知这种注解方式mailUsername、mailPassword、mailHost的值都是null,也就是直接给静态变量读取配置文件是无效的,如下所示img

方案一:

若要给静态变量赋值,可以使用set()方法,其中需要在类上加入@Component注解,方法名(例如setMailUsername)和参数名(例如username)可以任意命名,如下所示

调试结果如下:

img

@Component
public class JDConfig {
 
 
    /** 转换系统地址 */
 
    public static String url;
    /** 转换系统应用系统id */
 
    public static String sysId;
    /** 是否开启鉴权 */
 
    public static Boolean isAuth;
    /** 转换系统应用系统秘钥(如开启鉴权需要填写) */
 
    public static String sysKey;
 
    @Autowired(required = false)
    @Value(value="${jd.serverHost:}")
    public void setUrl( String url) {
        JDConfig.url = url;
    }
 
    @Autowired(required = false)
    @Value(value="${contract.jd.appKey:}")
    public void setSysId( String sysId) {
        JDConfig.sysId = sysId;
    }
    @Autowired(required = false)
    @Value(value="${jd.isAuth:true}")
    public void setAuth(Boolean isAuth) {
        JDConfig.isAuth = isAuth;
    }
 
    @Autowired(required = false)
    @Value(value="${contract.jd.appSecurity:}")
    public void setSysKey(String sysKey) {
        JDConfig.sysKey = sysKey;
    }
 
    public  String getUrl() {
        return url;
    }
 
    public  String getSysId() {
        return sysId;
    }
 
    public  Boolean getIsAuth() {
        return isAuth;
    }
 
    public  String getSysKey() {
        return sysKey;
    }
}

方案

如果你觉得@value注解麻烦。可以使用@ConfigurationProperties注解代替,这样比较简洁

  1. 前缀要写合适
  2. 方法名(例如setOssUrl)必须和属性保持一致,例如写为setUrl()会注入失败
  3. 类上加入@Component注解

最近的项目还有有这样一个需求就是类中有几个静态变量,初始化时候他们的值需要读取一个配置文件,获取一个code,然后这个code拼接而成。 这个code不是静态的变量,怎么实现的呢,代码如下:

    @Value("${projectCode}")
    private String projectCode;
 
    public static String COOPERATIVE_GOV_TEMPLATE_KEY ;
    // 消息短信配置
    public static String DEPOLY_KEY;
    // 消息短信详情配置
    public static String MSG_DEPOLY_KEY;
    // 过滤配置
    public static String MSG_FILTER_KEY;
 
    @PostConstruct
    public void init() {
        COOPERATIVE_GOV_TEMPLATE_KEY = projectCode + ":template";
        DEPOLY_KEY = projectCode + ":depoly";
        MSG_DEPOLY_KEY = projectCode + ":msgDepoly";
        MSG_FILTER_KEY = projectCode + ":msgFilter";
    }

这样当项目启动时候,这几个静态变量就有有值了。 一定要注意这个类要被spring管理,也就是要用@Controller,@Service,@Component等注解注释

方案三

  1. 类上加入@Component注解
  2. @PostConstruct注解修饰的方法中进行赋值操作

img

使用场景

那么问题来啦!我们什么场景需要把值注入到静态变量?

场景

img

场景

工具类中将值注入静态变量,就可以直接静态方法之中使用,我本文遇到的正是这个场景



@Value需要在Spring管理类中使用

 @Component
 public class FileComponent {
       public static Boolean enabledInline;
 	  public static String inlineWebUrl;
 	  @Value("${file.upload.enabledInline}")
       public void setEnabledInline(Boolean enabledInline) {
    		this.enabledInline = enabledInline;
       }
 	  @Value("${file.upload.inlineUrl}")
 	  public void setInlineWebUrl(String inlineWebUrl) {
    		this.inlineWebUrl = inlineWebUrl;
 	  }
  }

原文地址:https://blog.csdn.net/qq_43842093/article/details/130173219

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_43514.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注