我的编程空间,编程开发者的网络收藏夹
学习永远不晚

SpringBoot2动态@Value的实现方法

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

SpringBoot2动态@Value的实现方法

这篇文章主要介绍“SpringBoot2动态@Value的实现方法”,在日常操作中,相信很多人在SpringBoot2动态@Value的实现方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”SpringBoot2动态@Value的实现方法”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

前言

前面文章有详细描述过各个不同阶段对于bean的扩展接口

所以今天就基于BeanPostProcessor实现Spring中的@Value注解值动态变化

基于上面也可以实现一个配置中心,比如说Apollo

具体的实现步骤分为如下几步

通过BeanPostProcessor取得有使用@Value注解的bean,并存储到map中

动态修改map中的bean字段的值

获取bean

首先写一个类实现BeanPostProcessor接口,只需要使用其中的一个函数就可以。前后都可以用来实现,并不影响最终的使用,因为咱们只是需要bean的实例。

接下来看一下具体实现代码

package com.allen.apollo;import org.springframework.beans.BeansException;import org.springframework.beans.factory.annotation.Value;import org.springframework.beans.factory.config.BeanPostProcessor;import org.springframework.context.annotation.Configuration;import org.springframework.util.ReflectionUtils;import java.lang.reflect.Field;import java.util.LinkedList;import java.util.List;import java.util.Set;@Configurationpublic class SpringValueProcessor implements BeanPostProcessor {    private final PlaceholderHelper placeholderHelper = new PlaceholderHelper();    @Override    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {        if (beanName.equals("springValueController")) {            Class obj = bean.getClass();            List<Field> fields = findAllField(obj);            for (Field field : fields) {                Value value = field.getAnnotation(Value.class);                if (value != null) {                    Set<String> keys = placeholderHelper.extractPlaceholderKeys(value.value());                    for (String key : keys) {                        SpringValue springValue = new SpringValue(key, value.value(), bean, beanName, field, false);                        SpringValueCacheMap.map.put(key, springValue);                    }                }            }        }        return bean;    }    private List<Field> findAllField(Class clazz) {        final List<Field> res = new LinkedList<>();        ReflectionUtils.doWithFields(clazz, new ReflectionUtils.FieldCallback() {            @Override            public void doWith(Field field) throws IllegalArgumentException, IllegalAccessException {                res.add(field);            }        });        return res;    }}

上面的代码咱们就已经拿到了SpringValueController这个实例bean并存储到了map当中,下面看一下测试代码

  package com.allen.apollo;import com.google.common.collect.LinkedListMultimap;import com.google.common.collect.Multimap;public class SpringValueCacheMap {    public static final Multimap<String, SpringValue> map = LinkedListMultimap.create();}
 package com.allen.apollo;import java.lang.ref.WeakReference;import java.lang.reflect.Field;import java.lang.reflect.InvocationTargetException;import java.lang.reflect.Method;import java.lang.reflect.Type;import org.springframework.core.MethodParameter;public class SpringValue {    private MethodParameter methodParameter;    private Field field;    private WeakReference<Object> beanRef;    private String beanName;    private String key;    private String placeholder;    private Class<?> targetType;    private Type genericType;    private boolean isJson;    public SpringValue(String key, String placeholder, Object bean, String beanName, Field field, boolean isJson) {        this.beanRef = new WeakReference<>(bean);        this.beanName = beanName;        this.field = field;        this.key = key;        this.placeholder = placeholder;        this.targetType = field.getType();        this.isJson = isJson;        if (isJson) {            this.genericType = field.getGenericType();        }    }    public SpringValue(String key, String placeholder, Object bean, String beanName, Method method, boolean isJson) {        this.beanRef = new WeakReference<>(bean);        this.beanName = beanName;        this.methodParameter = new MethodParameter(method, 0);        this.key = key;        this.placeholder = placeholder;        Class<?>[] paramTps = method.getParameterTypes();        this.targetType = paramTps[0];        this.isJson = isJson;        if (isJson) {            this.genericType = method.getGenericParameterTypes()[0];        }    }    public void update(Object newVal) throws IllegalAccessException, InvocationTargetException {        if (isField()) {            injectField(newVal);        } else {            injectMethod(newVal);        }    }    private void injectField(Object newVal) throws IllegalAccessException {        Object bean = beanRef.get();        if (bean == null) {            return;        }        boolean accessible = field.isAccessible();        field.setAccessible(true);        field.set(bean, newVal);        field.setAccessible(accessible);    }    private void injectMethod(Object newVal)            throws InvocationTargetException, IllegalAccessException {        Object bean = beanRef.get();        if (bean == null) {            return;        }        methodParameter.getMethod().invoke(bean, newVal);    }    public String getBeanName() {        return beanName;    }    public Class<?> getTargetType() {        return targetType;    }    public String getPlaceholder() {        return this.placeholder;    }    public MethodParameter getMethodParameter() {        return methodParameter;    }    public boolean isField() {        return this.field != null;    }    public Field getField() {        return field;    }    public Type getGenericType() {        return genericType;    }    public boolean isJson() {        return isJson;    }    boolean isTargetBeanValid() {        return beanRef.get() != null;    }    @Override    public String toString() {        Object bean = beanRef.get();        if (bean == null) {            return "";        }        if (isField()) {            return String                    .format("key: %s, beanName: %s, field: %s.%s", key, beanName, bean.getClass().getName(), field.getName());        }        return String.format("key: %s, beanName: %s, method: %s.%s", key, beanName, bean.getClass().getName(),                methodParameter.getMethod().getName());    }}
package com.allen.apollo;import com.google.common.base.Strings;import com.google.common.collect.Sets;import org.springframework.beans.factory.config.BeanDefinition;import org.springframework.beans.factory.config.BeanExpressionContext;import org.springframework.beans.factory.config.ConfigurableBeanFactory;import org.springframework.beans.factory.config.Scope;import org.springframework.util.StringUtils;import java.util.Set;import java.util.Stack;public class PlaceholderHelper {  private static final String PLACEHOLDER_PREFIX = "${";  private static final String PLACEHOLDER_SUFFIX = "}";  private static final String VALUE_SEPARATOR = ":";  private static final String SIMPLE_PLACEHOLDER_PREFIX = "{";  private static final String EXPRESSION_PREFIX = "#{";  private static final String EXPRESSION_SUFFIX = "}";    public Object resolvePropertyValue(ConfigurableBeanFactory beanFactory, String beanName, String placeholder) {    // resolve string value    String strVal = beanFactory.resolveEmbeddedValue(placeholder);    BeanDefinition bd = (beanFactory.containsBean(beanName) ? beanFactory        .getMergedBeanDefinition(beanName) : null);    // resolve expressions like "#{systemProperties.myProp}"    return evaluateBeanDefinitionString(beanFactory, strVal, bd);  }  private Object evaluateBeanDefinitionString(ConfigurableBeanFactory beanFactory, String value,                                              BeanDefinition beanDefinition) {    if (beanFactory.getBeanExpressionResolver() == null) {      return value;    }    Scope scope = (beanDefinition != null ? beanFactory        .getRegisteredScope(beanDefinition.getScope()) : null);    return beanFactory.getBeanExpressionResolver()        .evaluate(value, new BeanExpressionContext(beanFactory, scope));  }    public Set<String> extractPlaceholderKeys(String propertyString) {    Set<String> placeholderKeys = Sets.newHashSet();    if (!isNormalizedPlaceholder(propertyString) && !isExpressionWithPlaceholder(propertyString)) {      return placeholderKeys;    }    Stack<String> stack = new Stack<>();    stack.push(propertyString);    while (!stack.isEmpty()) {      String strVal = stack.pop();      int startIndex = strVal.indexOf(PLACEHOLDER_PREFIX);      if (startIndex == -1) {        placeholderKeys.add(strVal);        continue;      }      int endIndex = findPlaceholderEndIndex(strVal, startIndex);      if (endIndex == -1) {        // invalid placeholder?        continue;      }      String placeholderCandidate = strVal.substring(startIndex + PLACEHOLDER_PREFIX.length(), endIndex);      // ${some.key:other.key}      if (placeholderCandidate.startsWith(PLACEHOLDER_PREFIX)) {        stack.push(placeholderCandidate);      } else {        // some.key:${some.other.key:100}        int separatorIndex = placeholderCandidate.indexOf(VALUE_SEPARATOR);        if (separatorIndex == -1) {          stack.push(placeholderCandidate);        } else {          stack.push(placeholderCandidate.substring(0, separatorIndex));          String defaultValuePart =              normalizeToPlaceholder(placeholderCandidate.substring(separatorIndex + VALUE_SEPARATOR.length()));          if (!Strings.isNullOrEmpty(defaultValuePart)) {            stack.push(defaultValuePart);          }        }      }      // has remaining part, e.g. ${a}.${b}      if (endIndex + PLACEHOLDER_SUFFIX.length() < strVal.length() - 1) {        String remainingPart = normalizeToPlaceholder(strVal.substring(endIndex + PLACEHOLDER_SUFFIX.length()));        if (!Strings.isNullOrEmpty(remainingPart)) {          stack.push(remainingPart);        }      }    }    return placeholderKeys;  }  private boolean isNormalizedPlaceholder(String propertyString) {    return propertyString.startsWith(PLACEHOLDER_PREFIX) && propertyString.endsWith(PLACEHOLDER_SUFFIX);  }  private boolean isExpressionWithPlaceholder(String propertyString) {    return propertyString.startsWith(EXPRESSION_PREFIX) && propertyString.endsWith(EXPRESSION_SUFFIX)        && propertyString.contains(PLACEHOLDER_PREFIX);  }  private String normalizeToPlaceholder(String strVal) {    int startIndex = strVal.indexOf(PLACEHOLDER_PREFIX);    if (startIndex == -1) {      return null;    }    int endIndex = strVal.lastIndexOf(PLACEHOLDER_SUFFIX);    if (endIndex == -1) {      return null;    }    return strVal.substring(startIndex, endIndex + PLACEHOLDER_SUFFIX.length());  }  private int findPlaceholderEndIndex(CharSequence buf, int startIndex) {    int index = startIndex + PLACEHOLDER_PREFIX.length();    int withinNestedPlaceholder = 0;    while (index < buf.length()) {      if (StringUtils.substringMatch(buf, index, PLACEHOLDER_SUFFIX)) {        if (withinNestedPlaceholder > 0) {          withinNestedPlaceholder--;          index = index + PLACEHOLDER_SUFFIX.length();        } else {          return index;        }      } else if (StringUtils.substringMatch(buf, index, SIMPLE_PLACEHOLDER_PREFIX)) {        withinNestedPlaceholder++;        index = index + SIMPLE_PLACEHOLDER_PREFIX.length();      } else {        index++;      }    }    return -1;  }}
package com.allen.apollo;import lombok.extern.slf4j.Slf4j;import org.springframework.beans.factory.annotation.Value;import org.springframework.util.StringUtils;import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.RestController;import java.lang.reflect.InvocationTargetException;@RestController@Slf4jpublic class SpringValueController {    @Value("${test:123}")    public String zax;    @Value("${test:123}")    public String test;    @Value(("${zed:zed}"))    public String zed;    @GetMapping("/test")    public String test(String a, String b) {        if (!StringUtils.isEmpty(a)) {            try {                for (SpringValue springValue : SpringValueCacheMap.map.get("test")) {                    springValue.update(a);                }                for (SpringValue springValue : SpringValueCacheMap.map.get("zed")) {                    springValue.update(b);                }            } catch (IllegalAccessException | InvocationTargetException e) {                e.printStackTrace();            }        }        return String.format("test: %s, zax: %s, zed: %s", test, zax, zed);    }}

到此,关于“SpringBoot2动态@Value的实现方法”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

SpringBoot2动态@Value的实现方法

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

SpringBoot2动态@Value的实现方法

这篇文章主要介绍“SpringBoot2动态@Value的实现方法”,在日常操作中,相信很多人在SpringBoot2动态@Value的实现方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”SpringBoo
2023-06-20

SpringBoot2动态@Value怎么实现

本篇内容介绍了“SpringBoot2动态@Value怎么实现”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!具体的实现步骤分为如下几步1.通
2023-06-08

java实现动态编译并动态加载的方法

小编给大家分享一下java实现动态编译并动态加载的方法,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!在D盘test目录下有个java文件:AlTest.javap
2023-06-14

java动态代理的实现方法

这篇文章主要介绍“java动态代理的实现方法”,在日常操作中,相信很多人在java动态代理的实现方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”java动态代理的实现方法”的疑惑有所帮助!接下来,请跟着小编
2023-06-20

SpringBoot2自动配置底层的方法

今天小编给大家分享一下SpringBoot2自动配置底层的方法的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1 自动配置底层
2023-06-29

Android滑动动态分页实现方法

本文实例讲述了Android滑动动态分页实现方法。分享给大家供大家参考,具体如下: 实现 Android.widget.AbsListView.OnScrollListener 主要代码:private int lastItemIndex;
2022-06-06

opencv检测动态物体的实现方法

这篇文章主要介绍“opencv检测动态物体的实现方法”,在日常操作中,相信很多人在opencv检测动态物体的实现方法问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”opencv检测动态物体的实现方法”的疑惑有所
2023-06-20

SpringBoot2整合Redis实现读写操作的方法

本篇内容主要讲解“SpringBoot2整合Redis实现读写操作的方法”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“SpringBoot2整合Redis实现读写操作的方法”吧!目录1. 启动
2023-06-20

pytorch动态神经网络的实现方法

这篇文章主要介绍了pytorch动态神经网络的实现方法,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。(1)首先要建立数据集import torch #引用torch模块im
2023-06-14

利用java实现动态代理的方法

这篇文章将为大家详细讲解有关利用java实现动态代理的方法,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。java 动态代理的方法总结AOP的拦截功能是由java中的动态代理来实现的。说白了,
2023-05-31

golangjsoniterextension处理动态字段的实现方法

这篇文章主要介绍了golangjsoniterextension处理动态字段的实现方法,我们使用实例级别的extension,而非全局,可以针对不同业务逻辑有所区分,jsoniter包提供了比较完善的定制能力,通过例子可以感受一下扩展性,需要的朋友可以参考下
2023-05-14

ASP动态include文件284607实现方法

经常有这样的要求,根据不同的需求要求include不同的文件如各个人的不同设置,所以要求能动态include文件
2023-05-20

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录