自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Nacos 中配置 Map 類(lèi)型,不香!

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

大家好,我是君哥。

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

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

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

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

下面就來(lái)分享一下配置 Map 類(lèi)型的過(guò)程中遇到的問(wèn)題。

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 類(lèi)型,但一個(gè)致命的槽點(diǎn)就是 mapping 這個(gè) bean 不能自動(dòng)刷新。這樣如果修改了 Nacos 中配置,要想讓配置生效,就必須重啟應(yīng)用服務(wù),這怎么能接受呢?

2.ConfigurationProperties

2.1 使用方式

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

@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 類(lèi)型,但是跟第一種方式一樣,定義的 Map 類(lèi)型變量不能自動(dòng)刷新。

3.使用監(jiān)聽(tīng)

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

@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 類(lèi)型的配置,然后把配置 put 到本地變量 map。這個(gè)也可以完成我們的需求,但是有幾點(diǎn)需要注意。

3.1 服務(wù)重啟

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

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

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

3.2 并發(fā)問(wèn)題

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

4.改進(jìn)

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

4.1 使用字符串

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

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;
}

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

4.2 刷新本地 Map

把解析字符串的結(jié)果放到本地變量 map 上,考慮到 Nacos 中配置可能會(huì)發(fā)生變化,用定時(shí)線(xiàn)程池每 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è)寫(xiě)法的好處是不用每次調(diào)用都解析字符串,而是由異步線(xiàn)程每秒鐘刷新。但是也有兩個(gè)問(wèn)題:

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

5.總結(jié)

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

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

責(zé)任編輯:武曉燕 來(lái)源: 君哥聊技術(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-10-12 09:48:55

SSR JSPPHP

2020-09-25 15:50:41

鴻蒙小米國(guó)產(chǎn)

2020-10-21 09:19:27

Flutter開(kāi)源項(xiàng)目

2021-12-05 23:17:18

iOS蘋(píng)果系統(tǒng)

2021-07-08 06:52:41

ESClickHouse Lucene

2021-12-03 10:46:49

ELKGraylog運(yùn)維

2020-01-21 21:15:16

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

2021-12-02 06:34:34

GraylogELK日志

2021-04-06 10:48:52

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

2020-07-03 15:10:35

Java Rust 開(kāi)發(fā)

2021-05-19 09:37:45

SessionTokencookie

2020-04-28 10:17:51

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

2021-06-27 17:33:51

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

2021-11-29 06:24:05

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

2024-04-12 12:36:06

JSJavaScrip方式

2022-03-22 09:20:57

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

2020-07-30 09:10:21

DockerK8s容器
點(diǎn)贊
收藏

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