前言
本文的素材来源于敌人整合nacos2作为配置核心进行动静刷新时,踩到的坑。他过后遇到的问题,如下截图
因为那段时间比较忙,于是我在没看敌人我的项目代码的根底上,就找个了看似解决方案的答案,扔了过来
前面敌人加了这个配置,问题果然没有解决。前面就抽了一点工夫,要了他的我的项目代码来看下。
代码示例
因为他这个我的项目次要是他自学nacos的我的项目,也没波及啥敏感信息。本文就间接拿他的我的项目示例演示
1、我的项目pom依赖
<properties> <maven.compiler.source>8</maven.compiler.source> <maven.compiler.target>8</maven.compiler.target> <java.version>1.8</java.version> <spring-boot.version>2.3.12.RELEASE</spring-boot.version> <spring-cloud.version>Hoxton.SR12</spring-cloud.version> <spring-cloud-alibaba.version>2.2.8.RELEASE</spring-cloud-alibaba.version> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> </dependencies> <dependencyManagement> <dependencies> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-alibaba-dependencies</artifactId> <version>${spring-cloud-alibaba.version}</version> <type>pom</type> <scope>import</scope> </dependency> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-dependencies</artifactId> <version>${spring-boot.version}</version> <type>pom</type> <scope>import</scope> </dependency> <dependency> <groupId>org.springframework.cloud</groupId> <artifactId>spring-cloud-dependencies</artifactId> <version>${spring-cloud.version}</version> <type>pom</type> <scope>import</scope> </dependency> </dependencies> </dependencyManagement> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <dependency> <groupId>com.alibaba.cloud</groupId> <artifactId>spring-cloud-starter-alibaba-nacos-config</artifactId> </dependency> </dependencies>
注: nacos服务端版本为2.1.1
2、nacos配置核心地址,配置在bootstrap.yml外面
spring: cloud: nacos: server-addr: localhost:8848
3、我的项目的根本信息配置在application.yml外面
#spring: application: name: nacos-configserver: port: 8030
4、编写一个须要动静刷新获取值的controller
@RestController@RequestMapping("/config")@RefreshScopepublic class ConfigController { @Value("${user.userName:123}") private String userName; @RequestMapping("/get") private String get(){ return userName; }}
5、业务我的项目在nacos服务端上配置如下
以上就是敌人残缺的代码例子。咱们运行一下代码,会发现controller的userName取不到值。感兴趣的敌人,能够走查一下上述的代码,查找一下起因
取不到值的起因
理论知识铺垫:
当咱们应用cglib动静代理调用指标办法时,当办法被private润饰时,this为动静代理对象。当办法被public或者protected润饰时,this为指标对象。此外属性刷新刷的是指标对象的属性,controller的get办法能够看成是
@RequestMapping("/get") private String get(){ return this.userName; }
当咱们在controller加上@RefreshScope注解时,如果不扭转变proxyMode这个属性值时,他默认就会生成一个cglib动静代理。当咱们调用get办法,因为get为公有办法,咱们能够看成
@RequestMapping("/get") private String get(){ return cglibProxy.userName; }
此时的this是代理对象,而此时userName是代理对象的userName,代理对象的userName是空值。
解决办法
办法一、批改@RefreshScope的proxyMode属性
将proxyMode改为ScopedProxyMode.DEFAULT或者ScopedProxyMode.NO
此时this为指标对象,因而能取到值
办法二:将指标办法的修饰符改为public或者protected
示例:
此时this为指标对象,因而能取到值
3、办法三:应用属性配置类
@Configuration@ConfigurationProperties(prefix = "user")public class UserProperties { private String userName; public String getUserName() { return userName; } public void setUserName(String userName) { this.userName = userName; }}
controller调整成
@RestController@RequestMapping("/config")public class ConfigController { @Autowired private UserProperties userProperties; @RequestMapping("/get") public String get(){ return userProperties.getUserName(); } }
此时controller不必加@RefreshScope也能实现动静刷新。因为属性类上的@ConfigurationProperties自身就具备动静刷新的个性
总结
本文不算是@RefreshScope与nacos2整合踩到的坑,次要还是动静代理方面的常识,题目有点题目党了。
有些视频讲nacos动静刷新时,基本上都是举controller上@RerfreshScope +@value来解说。其实利用@ConfigurationProperties也是能够达到相似的成果。如果没和springcloud整合,引入nacos配置核心的starter,应用@NacosPropertySource + @NacosValue或者@NacosRefresh也是能够实现动静刷新,感兴趣的敌人能够试一下
最初,敌人之前在nacos2搭建过程中,也踩到了一些坑。感兴趣的敌人能够查看如下文章
记一次应用nacos2踩到的坑