成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Nacos 中配置 Map 類型,不香!

開發(fā) 前端
Nacos API 提供了監(jiān)聽功能,可以監(jiān)聽配置的變化,對變化進(jìn)行處理,只要在監(jiān)聽方法上增加 @NacosConfigListener 這個(gè)注解就可以生效。

大家好,我是君哥。

最近在使用 Nacos 過程中遇到一個(gè)場景,配置的字符串可以解析成 Map 類型使用,有一個(gè)配置如下:

map:
test: key1:value1,key2:value2,key3:value3

后來有同事建議 Nacos 可以直接配置成 Map 類型,后臺(tái)使用 Java Map 類型獲取就可以。配置如下:

map:
test:
key1: value1
key2: value2
key3: value3

下面就來分享一下配置 Map 類型的過程中遇到的問題。

1.使用 Bean 方式獲取配置

1.1 使用方式

參考網(wǎng)上的一些案例,第一個(gè)方式是把讀取到的 Map 作為一個(gè) Spring 的 Bean,一看代碼就明白了。

@Bean
@ConfigurationProperties(prefix = "map.test")
public Map<String, String> mapping() {
return new HashMap<>();
}

1.2 槽點(diǎn)

這樣確實(shí)可以把 Nacos 中讀取到的配置轉(zhuǎn)換成 Map 類型,但一個(gè)致命的槽點(diǎn)就是 mapping 這個(gè) bean 不能自動(dòng)刷新。這樣如果修改了 Nacos 中配置,要想讓配置生效,就必須重啟應(yīng)用服務(wù),這怎么能接受呢?

2.ConfigurationProperties

2.1 使用方式

直接使用 @Value 和 @NacosValue 是獲取不到值的。下面的這種方式,類的定義上加注解 @ConfigurationProperties,再定義一個(gè)變量,名稱跟 Nacos 中配置的后綴一樣,這樣是可以獲取到 Map 類型的配置的。

@Component
@RefreshScope
@ConfigurationProperties(prefix = "map")
public class NacosRefresh {

private Logger logger = LoggerFactory.getLogger(getClass());

public void setTest(Map<String, String> test) {
this.test = test;
}

private Map<String, String> test;
}

注意:上面的 setTest 方法是必須要的,不然 test 變量取不到值。

2.2 槽點(diǎn)

這樣確實(shí)可以把 Nacos 中讀取到的配置轉(zhuǎn)換成 Map 類型,但是跟第一種方式一樣,定義的 Map 類型變量不能自動(dòng)刷新。

3.使用監(jiān)聽

Nacos API 提供了監(jiān)聽功能,可以監(jiān)聽配置的變化,對變化進(jìn)行處理,只要在監(jiān)聽方法上增加 @NacosConfigListener 這個(gè)注解就可以生效。見下面代碼:

@Service
public class NacosListener {

private Logger logger = LoggerFactory.getLogger(getClass());

private Map<String, String> map = new HashMap<>();

@NacosConfigListener(dataId = "maptest.yaml",groupId = "DEFAULT_GROUP")
public void listener(String context){
logger.info("================listener context:{}", context);
if (StringUtils.isBlank(context)){
return;
}
Yaml yaml = new Yaml();
Map<String, Object> contextMap = yaml.load(context);
Map<String, Object> map = (Map<String, Object>)contextMap.get("map");
if (CollectionUtils.isEmpty(map)){
return;
}
Map<String, String> test = (Map<String, String>) map.get("test");
if (CollectionUtils.isEmpty(test)){
return;
}
map.clear();
map.putAll(test);
map.forEach((k,v) -> logger.info("Entry in map, key:{},value:{}", k, v));
}
}

這段代碼是從 Nacos 配置中解析出 Map 類型的配置,然后把配置 put 到本地變量 map。這個(gè)也可以完成我們的需求,但是有幾點(diǎn)需要注意。

3.1 服務(wù)重啟

如果服務(wù)重啟了,本地變量 map 拉不到值。因?yàn)樯厦姹O(jiān)聽的邏輯并沒有走,即使在 Nacos 上重新發(fā)布一下,也不行。

上面的監(jiān)聽方法,只有在 Nacos 配置發(fā)生變化并且發(fā)布后才會(huì)觸發(fā),比如 map.test 配置改變?nèi)缦拢?/p>

map:
test:
key1: value1
key2: value2
key3: value3
key4: value4

3.2 并發(fā)問題

上面監(jiān)聽的代碼里面,需要把本地變量 map 先 clear 然后再 putAll,如果這兩個(gè)方法調(diào)用中間發(fā)生了線程上下文切換,讀取線程可能會(huì)因?yàn)閺?map 中取不到值而發(fā)生異常。

4.改進(jìn)

上面講解了使用 Nacos 配置 Map 類型的坑,不過使用 Nacos 配置 Map 類型也有個(gè)好處,不用解析字符串,直接可以轉(zhuǎn)成 Map 類型。

4.1 使用字符串

完全不使用 Map 類型了,改成配置字符串,配置如下:

map:
test: key1:value1,key2:value2,key3:value3

解析代碼如下:

@NacosValue(value = "${map.test}", autoRefreshed = true)
private String mapTest;

public String get(String key){
String[] keys = mapTest.split(",");
for (String item : keys){
if (!item.contains(key)){
continue;
}
return item.split(":")[1];
}
return null;
}

這種寫法的好處是不用監(jiān)聽 Nacos,配置改變后 mapTest 變量自動(dòng)刷新,缺點(diǎn)是每次調(diào)用 get 方法都需要解析 mapTest 這個(gè)字符串。

4.2 刷新本地 Map

把解析字符串的結(jié)果放到本地變量 map 上,考慮到 Nacos 中配置可能會(huì)發(fā)生變化,用定時(shí)線程池每 1 秒刷新一次,代碼如下:

private Map<String, String> map = new HashMap<>();

@NacosValue(value = "${map.test}", autoRefreshed = true)
private String mapTest;

@PostConstruct
public void refreshLocalMap(){
ScheduledThreadPoolExecutor scheduled = new ScheduledThreadPoolExecutor(1);
scheduled.scheduleAtFixedRate(() -> refresh(), 0, 1000, TimeUnit.MILLISECONDS);
}

public void refresh(){
String[] keys = mapTest.split(",");
for (String item : keys){
String[] kv = item.split(":");
map.put(kv[0], kv[1]);
}
}

這個(gè)寫法的好處是不用每次調(diào)用都解析字符串,而是由異步線程每秒鐘刷新。但是也有兩個(gè)問題:

  • 需要一個(gè)定時(shí)線程池,會(huì)消耗 CPU 資源。
  • refresh 方法是每秒執(zhí)行一次,會(huì)有短暫的本地變量和 Nacos 配置不一致的問題。

5.總結(jié)

Nacos 中配置 Map 類型確實(shí)不香,主要原因是刷新不方便。但是對于配置不需要刷新的場景,還是很有好處的,尤其是 key 比較多的時(shí)候,比解析字符串方便很多,而且 Hash 的時(shí)間復(fù)雜度是 o(1) ,在數(shù)據(jù)結(jié)構(gòu)中是最優(yōu)秀的。

對于需要刷新的場景,無論使用哪種方案,都有優(yōu)缺點(diǎn),沒有最好的,只有最適合的,要根據(jù)系統(tǒng)的業(yè)務(wù)場景來做選擇。

責(zé)任編輯:武曉燕 來源: 君哥聊技術(shù)
相關(guān)推薦

2020-05-25 10:37:58

自學(xué)編程技巧

2021-01-11 08:03:30

阿里中臺(tái)項(xiàng)目

2021-12-17 15:05:55

CSSwhenelse

2020-09-25 15:50:41

鴻蒙小米國產(chǎn)

2020-10-12 09:48:55

SSR JSPPHP

2021-12-05 23:17:18

iOS蘋果系統(tǒng)

2020-10-21 09:19:27

Flutter開源項(xiàng)目

2021-07-08 06:52:41

ESClickHouse Lucene

2021-12-03 10:46:49

ELKGraylog運(yùn)維

2021-12-02 06:34:34

GraylogELK日志

2020-01-21 21:15:16

WiFi網(wǎng)絡(luò)WiFi6

2021-04-06 10:48:52

MySQLElasticsear數(shù)據(jù)庫

2025-06-25 09:31:41

2020-07-03 15:10:35

Java Rust 開發(fā)

2021-05-19 09:37:45

SessionTokencookie

2021-06-27 17:33:51

培訓(xùn)員工AI人工智能

2020-04-28 10:17:51

人工智能技術(shù)智能音箱

2021-11-29 06:24:05

物聯(lián)網(wǎng)平臺(tái)物聯(lián)網(wǎng)IOT

2022-03-22 09:20:57

應(yīng)用線程池技術(shù)

2021-11-02 16:44:40

部署DevtoolsJRebel
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號

主站蜘蛛池模板: 亚洲影音 | 极品国产视频 | h在线| 国产精品美女久久久免费 | 精品一区二区三区在线观看国产 | 日韩一二区在线 | 天堂一区二区三区四区 | 国产一区二区精品在线 | 免费毛片网 | 亚洲美女网站 | 97精品超碰一区二区三区 | 日韩av成人| 91婷婷韩国欧美一区二区 | 亚洲三区在线观看 | 国产精品福利在线 | 超碰人人在线 | 午夜影院在线观看免费 | 日韩一区二区三区在线看 | 欧美一区二区三区精品免费 | 国产精品99| 天天操综合网站 | 国产精品久久久久久久7电影 | 黄色a三级 | 亚洲精品一 | 亚洲视频中文 | 国产黄色av电影 | 91国在线高清视频 | 亚洲一级二级三级 | 日韩中文字幕在线视频 | 国产精品视频久久 | 国产精品毛片一区二区在线看 | 欧美成人一区二免费视频软件 | 亚洲在线一区二区 | 成年网站在线观看 | 91视频播放 | 亚洲精品成人网 | 国产精品视频久久久 | 欧美极品在线视频 | 精品国产一区二区三区久久久蜜月 | 国产免费av网 | 91精品一区 |