最近公司要搭建一个配置中心,由于不想自己运维,便选择了云服务,定的是阿里云的nacos服务,即 https://mse.console.aliyun.com/。
这里记录一下在springboot 2项目中,使用云nacos来做一些基本配置String、int等,和配置json数据映射到java对象中。
在控制台创建一个实例,这个实例就是一个nacos的服务端。
在创建时,需要注意公网ip的问题,如果没有公网ip,就只能在阿里云服务器上才能访问,本地测试的话访问不了。
如图,基本信息里有公网、内网连接地址、端口信息。因为本篇是在本地使用云nacos,所以我在创建nacos时选择了外网地址。红框里需要注意配置白名单,不在白名单里的ip无法访问云nacos。本机的话,就在百度搜索ip,查看自己的外网ip,配进去如11.11.11.11/32即可。
当然配白名单麻烦,建议采用accessKey的方式访问配置中心,也就是把公网白名单删掉,把权限验证打开。然后用access-key和secret-key来访问nacos。
之后就可以创建配置了。
这里我没有创建命名空间,就默认是public的命名空间。
创建配置时,Group里填写名字,建议一个模块、一个应用用同一个Group,代表一个组。
Data ID的范围比Group要小,代表一个类型、一个对象,譬如我把系统里所有的开关都放到一个Data Id里,里面配很多个开关。譬如我可以把一个大对象,有多个属性,也作为一个Data ID。
将来监听变化,自动拉取最新配置信息的最小单元就是Data ID。
如图配置格式选择TEXT,就是最简单的类型,配置内容里就可以是a=1,b=2这种。这里面的每一行都是一个最小的配置项,将来代码里用的就是最小的配置项。
nacos提供了支持java、springboot的pom依赖,我们先来看看普通的java接入方式。
依赖的pom就不贴了,直接看使用的代码。
- import com.alibaba.nacos.api.NacosFactory;
- import com.alibaba.nacos.api.config.ConfigService;
- import com.alibaba.nacos.api.config.listener.Listener;
- import com.alibaba.nacos.api.exception.NacosException;
-
- import java.util.Properties;
- import java.util.concurrent.Executor;
- import java.util.concurrent.ExecutorService;
- import java.util.concurrent.Executors;
-
- /**
- * @author weifengwu
- * @create 2023/9/4 15:48
- */
- public class NacosTest {
- public static void main(String[] args) {
- try {
- ExecutorService executorService = Executors.newFixedThreadPool(5);
- //改成自己的配置外网地址
- String serverAddr = "mse-xxxxxxx-p.nacos-ans.mse.aliyuncs.com:8848";
- Properties properties = new Properties();
- properties.put("serverAddr", serverAddr);
- //所有对于配置项的操作都是靠ConfigService完成的
- ConfigService configService = NacosFactory.createConfigService(properties);
-
- //读取该dataId的内容。注意,每次调用getConfig都会发起一次网络请求
- String content = configService.getConfig( "switch", "llmCenter", 5000);
- System.out.println(content);
-
- //读取一个配置项,该配置项是个json字符串,
- // "app-pc-lu": {
- // "userAudit": 1,
- // "aiAudit": 14,
- // "modelList": [
- // "modelA",
- // "modelB",
- // "modelC"
- // ]
- // }
- String jsonAppFlowInfo = configService.getConfig("jsonAppFlowInfo", "llmCenter", 1000);
- //给这个Data ID添加一个监听器,当DataId数据发生变化时,会调用该回调方法。用户可以接收到新值后保存到内存里。
- configService.addListener("jsonAppFlowInfo", "llmCenter", new Listener() {
- @Override
- public Executor getExecutor() {
- return executorService;
- }
-
- @Override
- public void receiveConfigInfo(String s) {
- System.err.println(Thread.currentThread().getName());
- System.out.println(s);
- }
- });
- System.out.println(jsonAppFlowInfo);
-
- try {
- Thread.sleep(100000000);
- } catch (InterruptedException e) {
- throw new RuntimeException(e);
- }
-
- //新增一个配置,如果已存在该DataId,则会覆盖
- // boolean isPublishOk = false;
- // try {
- // isPublishOk = configService.publishConfig("second", "llmCenter", "content=1");
- // } catch (NacosException e) {
- // throw new RuntimeException(e);
- // }
- // System.out.println(isPublishOk);
-
- } catch (NacosException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
以上代码展示了获取配置、新增/修改配置,监听配置的功能。
需要注意的是configService.getConfig读取配置信息时,每次都会发起一个网络请求,并不是从本地内存读取的,这一点和我的认知中的配置中心不太一样,如果说的不对,可以指正,因为我从nacos的后台看到确实在不断的请求。
如果使用的是springboot,nacos提供了一些自动化监听的功能,当服务端配置的值发生变更时,不需要客户端做处理,即可自动读取到新的配置内容。
- @RestController
- @NacosPropertySource(groupId = "llmCenter", dataId = "switch", autoRefreshed = true)
- @NacosPropertySource(groupId = "llmCenter", dataId = "textInfo", autoRefreshed = true)
- public class TestController {
- @Autowired
- private ConvMessageService convMessageService;
-
- @NacosValue(value = "${useLocalCache:false}",autoRefreshed = true)
- private boolean useLocalCache;
-
- @NacosValue(value = "${one:1}",autoRefreshed = true)
- private String one;
-
- @NacosValue(value = "${model.name.app-pc-luca.userAudit:2}",autoRefreshed = true)
- private String userAudit;
-
- /**
- * 这种写法不支持,将无法映射
- */
- @NacosValue(value = "${model.name.app-pc-luca.modelList:2}",autoRefreshed = true)
- private List
modelList; -
- @RequestMapping(value = "/get", method = GET)
- public boolean get() {
- System.out.println("useLocalCache-" + useLocalCache);
- System.out.println("one-" + one);
- System.out.println("userAudit-" + userAudit);
- System.out.println("modelList-" + modelList);
-
- return true;
- }
- }
其中NacosPropertySource注解加在一个被spring托管的类上即可,里面注明了dataId和groupId,代表这个dataId的配置值发生变化时会自动推送到被@NacosValue标注的属性上。
其中@NacosValue里只需要写最终配置的属性名即可,不需要再指定dataId,也就是对应下图的每一行里的key。
@NacosValue(value = "${useLocalCache:false}",autoRefreshed = true)这个里面useLocalCache:false代表如果配置中心里没找到useLocalCache这个key,就用false作为默认值。
application.yml配置如下:
- nacos:
- config:
- server-addr: mse-80a0d732-p.nacos-ans.mse.aliyuncs.com:8848
- access-key: xx
- secret-key: xxx
- autoRefresh: true
- data-ids: switch,hello,textInfo
- group: llmCenter
- bootstrap:
- enable: true
其中需要注意的点是,yml里面要写data-ids,这里面写的data-ids是等同于在java类上加的@NacosPropertySource(groupId = "llmCenter", dataId = "switch", autoRefreshed = true)。二者功能相同。
如果你在yml里配置了data-ids,那么类上的可以不用写。个人建议是都写在yml文件里即可。
本身并不支持将一个json格式的文本,直接映射到一个java对象上。但如果你的配置中心里,确实是需要配一个json,那么建议用一个data-id来存放,里面放个json字符串。要使用时,自己解析即可。
见第二段java接入云nacos里的代码,通过下面的方式,获取到json字符串,并监听这个data-id字符串。
正确做法应该是,在项目启动时,就拉取配置,并将结果保存到系统内存中,使用时从内存中获取。当监听到变化时,也更新内存中的json对象。不要每次都用configService.getConfig获取。
- //读取一个配置项,该配置项是个json字符串,
- // "app-pc-lu": {
- // "userAudit": 1,
- // "aiAudit": 14,
- // "modelList": [
- // "modelA",
- // "modelB",
- // "modelC"
- // ]
- // }
- String jsonAppFlowInfo = configService.getConfig("jsonAppFlowInfo", "llmCenter", 1000);
- //给这个Data ID添加一个监听器,当DataId数据发生变化时,会调用该回调方法。用户可以接收到新值后保存到内存里。
- configService.addListener("jsonAppFlowInfo", "llmCenter", new Listener() {
- @Override
- public Executor getExecutor() {
- return executorService;
- }
-
- @Override
- public void receiveConfigInfo(String s) {
- System.err.println(Thread.currentThread().getName());
- System.out.println(s);
- }
- });
- System.out.println(jsonAppFlowInfo);