SpringBoot整合Apollo配置中心快速使用詳解
一、簡(jiǎn)介
1.Apollo 是什么?Apollo(阿波羅)是攜程框架部門(mén)研發(fā)的分布式配置中心。服務(wù)端基于Spring Boot和Spring Cloud開(kāi)發(fā)。
2.為什么要使用Apollo?
- 安全性:配置跟隨源代碼保存在代碼庫(kù)中,容易造成配置泄漏
- 時(shí)效性:普通方式配置,修改配置,需要重啟服務(wù)才能生效
- 局限性:無(wú)法支持動(dòng)態(tài)調(diào)整:例如日志開(kāi)關(guān)、功能開(kāi)關(guān)
二、使用
1. 測(cè)試項(xiàng)目搭建
注:本文主要介紹SpringBoot 整合 Apollo 實(shí)現(xiàn)動(dòng)態(tài)配置
1.1 添加Maven依賴(lài)
<dependency> <groupId>com.ctrip.framework.apollo</groupId> <artifactId>apollo-client</artifactId> <version>1.3.0</version> </dependency>
1.2 配置文件
# apollo集成 # apollo 配置應(yīng)用的 appid app.id=springboot-apollo-demo1 # apollo meta-server地址,一般同config-server地址 apollo.meta=http://192.168.0.153:8080 #啟用apollo配置開(kāi)關(guān) apollo.bootstrap.enabled=true apollo.bootstrap.eagerLoad.enabled=true # apollo 使用配置的命名空間,多個(gè)以逗號(hào)分隔 apollo.bootstrap.namespaces = application
配置說(shuō)明:
- app.id:在配置中心配置的應(yīng)用身份信息。
- apollo.bootstrap.enabled:在應(yīng)用啟動(dòng)階段是否向Spring容器注入被托管的properties文件配置信息。
- apollo.bootstrap.eagerLoad.enabled:將Apollo配置加載提到初始化日志系統(tǒng)之前。
- apollo.bootstrap.namespaces:配置的命名空間,多個(gè)逗號(hào)分隔,一個(gè)namespace相當(dāng)于一個(gè)配置文件。
- **apollo.meta:**當(dāng)前環(huán)境服務(wù)配置地址,生產(chǎn)環(huán)境建議至少雙節(jié)點(diǎn),可以填寫(xiě)多個(gè)逗號(hào)分隔,使用一個(gè)單獨(dú)的域,如 http://config.xxx.com(由nginx等軟件負(fù)載平衡器支持),而不是多個(gè)IP地址,因?yàn)榉?wù)器可能會(huì)擴(kuò)展或縮小。
圖示說(shuō)明:

1.3 添加啟動(dòng)類(lèi)
@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class})
public class SpringbootApolloApplication {
public static void main(String[] args) {
SpringApplication.run(SpringbootApolloApplication.class, args);
}
}
1.4 添加配置開(kāi)關(guān)類(lèi)
基于@Value注解配置
@Component
public class ValueStyleProperty {
@Value("${apollo.value.demoKey1}")
private String demoKey1;
@Value("${apollo.value.demoKey2}")
private String demoKey2;
//省略 get/set 方法
}
1.5 添加測(cè)試controller
/**
* value注解方式,獲取屬性
*
* @author mengqiang
*/
@RestController
@RequestMapping("/value-style")
public class ValuePropertyController {
@Autowired
private ValueStyleProperty keyProperty;
@Value("${server.port}")
private String port;
@Value("${apollo.bootstrap.namespaces:'application'}")
private String namespaces;
@GetMapping("/get")
public Map<String, Object> getProperty() {
Map<String, Object> map = new LinkedHashMap<>();
map.put("port", port);
map.put("namespaces", namespaces);
map.put("demoKey1", keyProperty.getDemoKey1());
map.put("demoKey2", keyProperty.getDemoKey2());
return map;
}
}
2. Apollo配置中心的配置
2.1 創(chuàng)建項(xiàng)目

2.2 填寫(xiě)配置信息

配置說(shuō)明:
- 部門(mén):選擇應(yīng)用所在的部門(mén)。(可自定義部門(mén))
- 應(yīng)用AppId:用來(lái)標(biāo)識(shí)應(yīng)用身份的唯一id,格式為string,需與application.properties中配置的app.id一致。
- 應(yīng)用名稱(chēng):應(yīng)用名,僅用于界面展示。
- 應(yīng)用負(fù)責(zé)人:選擇的人默認(rèn)會(huì)成為該項(xiàng)目的管理員,具備項(xiàng)目權(quán)限管理、集群創(chuàng)建、Namespace創(chuàng)建等權(quán)限。
項(xiàng)目配置主頁(yè)截圖

2.3 添加配置
2.3.1 表格形式單個(gè)添加
注:不能批量操作


2.3.2 文本形式批量添加
注:可實(shí)現(xiàn)批量操作



2.4 發(fā)布配置
注:配置只有發(fā)布后才會(huì)生效

點(diǎn)擊發(fā)布按鈕

2.5 多環(huán)境同步配置
注意事項(xiàng):
通過(guò)同步配置功能,可以使多個(gè)環(huán)境、集群間的配置保持一致需要注意的是,同步完之后需要發(fā)布后才會(huì)對(duì)應(yīng)用生效
點(diǎn)擊同步配置

選擇需要同步的配置,以及目標(biāo)環(huán)境

點(diǎn)擊同步

目標(biāo)環(huán)境查看

3. 項(xiàng)目啟動(dòng)與測(cè)試
3.1 初始啟動(dòng)讀取測(cè)試

3.2 自動(dòng)更新屬性測(cè)試

發(fā)布后控制臺(tái)變化

測(cè)試輸出值變化

4.常見(jiàn)整合問(wèn)題
4.1@ConfigurationProperties注解整合Apollo不生效問(wèn)題
示例配置類(lèi)
/**
* 公共開(kāi)關(guān),key值 屬性配置
*
* @author mengqiang
*/
@Component
@ConfigurationProperties(prefix = "apollo.first.config")
public class ConfigFirstProperty {
/**
* 測(cè)試數(shù)字
*/
private Integer oneNumber;
/**
* 測(cè)試字符串
*/
private String oneStr;
/**
* 啟用標(biāo)記
*/
private Boolean oneEnableFlag;
/**
* 稅率 默認(rèn) 0.03
*/
private BigDecimal oneTaxRate = new BigDecimal("0.03");
//省略 get/set 方法
}
解決方案
添加監(jiān)聽(tīng)配置
import com.ctrip.framework.apollo.model.ConfigChange;
import com.ctrip.framework.apollo.model.ConfigChangeEvent;
import com.ctrip.framework.apollo.spring.annotation.ApolloConfigChangeListener;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
import org.springframework.cloud.context.environment.EnvironmentChangeEvent;
/**
* Apollo 配置監(jiān)聽(tīng)
*/
@Configuration
public class ApolloConfigListener implements ApplicationContextAware {
/**
* 日志
*/
private static final Logger LOGGER = LoggerFactory.getLogger(ApolloConfigListener.class);
private ApplicationContext applicationContext;
/**
* 配置監(jiān)聽(tīng)
* ApolloConfigChangeListener > value 屬性默認(rèn) 命名空間 "application"
*
* 示例: @ApolloConfigChangeListener(value = {"application", "test_space"})
*/
@ApolloConfigChangeListener
private void onChange(ConfigChangeEvent changeEvent) {
LOGGER.info("【Apollo-config-change】start");
for (String key : changeEvent.changedKeys()) {
ConfigChange change = changeEvent.getChange(key);
LOGGER.info("key={} , propertyName={} , oldValue={} , newValue={} ", key, change.getPropertyName(), change.getOldValue(), change.getNewValue());
}
// 更新相應(yīng)的bean的屬性值,主要是存在@ConfigurationProperties注解的bean
this.applicationContext.publishEvent(new EnvironmentChangeEvent(changeEvent.changedKeys()));
LOGGER.info("【Apollo-config-change】end");
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
}
}
4.2日志級(jí)別未更新問(wèn)題
示例配置
logging.level.com.example=info
解決方案-日志監(jiān)聽(tīng)器
import com.ctrip.framework.apollo.model.ConfigChange;
import com.ctrip.framework.apollo.model.ConfigChangeEvent;
import com.ctrip.framework.apollo.spring.annotation.ApolloConfigChangeListener;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.logging.LogLevel;
import org.springframework.boot.logging.LoggingSystem;
import org.springframework.context.annotation.Configuration;
import javax.annotation.Resource;
/**
* Apollo 日志-配置監(jiān)聽(tīng)
*/
@Configuration
public class LoggerConfigListener {
private static final Logger LOGGER = LoggerFactory.getLogger(LoggerConfigListener.class);
private static final String LOGGER_TAG = "logging.level.";
@Resource
private LoggingSystem loggingSystem;
/**
* 監(jiān)聽(tīng) 日志配置的變化
*/
@ApolloConfigChangeListener(interestedKeyPrefixes = LOGGER_TAG)
private void onChangeLogger(ConfigChangeEvent changeEvent) {
LOGGER.info("【Apollo-logger-config-change】>> start");
refreshLoggingLevel(changeEvent);
LOGGER.info("【Apollo-logger-config-change】>> end");
}
/**
* 刷新日志級(jí)別
*/
private void refreshLoggingLevel(ConfigChangeEvent changeEvent) {
if (null == loggingSystem) {
return;
}
for (String key : changeEvent.changedKeys()) {
ConfigChange change = changeEvent.getChange(key);
if (!StringUtils.containsIgnoreCase(key, LOGGER_TAG)) {
continue;
}
LOGGER.info("【Apollo-logger-config-change】>> key={} , propertyName={} , oldValue={} , newValue={} ",
key, change.getPropertyName(), change.getOldValue(), change.getNewValue());
String newLevel = change.getNewValue();
LogLevel level = LogLevel.valueOf(newLevel.toUpperCase());
loggingSystem.setLogLevel(key.replace(LOGGER_TAG, ""), level);
LOGGER.info("【Apollo-logger-config-change】>> {} -> {}", key, newLevel);
}
}
}
4.3日志+配置類(lèi)自動(dòng)刷新整合監(jiān)聽(tīng)
注:由于 4.1與4.2監(jiān)聽(tīng)有重合,所以最好放在一起處理
import com.ctrip.framework.apollo.model.ConfigChange;
import com.ctrip.framework.apollo.model.ConfigChangeEvent;
import com.ctrip.framework.apollo.spring.annotation.ApolloConfigChangeListener;
import org.apache.commons.lang3.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.BeansException;
import org.springframework.boot.logging.LogLevel;
import org.springframework.boot.logging.LoggingSystem;
import org.springframework.cloud.context.environment.EnvironmentChangeEvent;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.context.annotation.Configuration;
import javax.annotation.Resource;
/**
* Apollo 配置監(jiān)聽(tīng)
*/
@Configuration
public class ApolloConfigListener implements ApplicationContextAware {
/**
* 日志
*/
private static final Logger LOGGER = LoggerFactory.getLogger(ApolloConfigListener.class);
/**
* 日志配置常量
*/
private static final String LOGGER_TAG = "logging.level.";
@Resource
private LoggingSystem loggingSystem;
private ApplicationContext applicationContext;
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
}
/**
* 配置監(jiān)聽(tīng)
* ApolloConfigChangeListener > value 屬性默認(rèn) 命名空間 "application"
*/
@ApolloConfigChangeListener
private void onChangeConfig(ConfigChangeEvent changeEvent) {
LOGGER.info("【Apollo-config-change】>> start");
for (String key : changeEvent.changedKeys()) {
ConfigChange change = changeEvent.getChange(key);
LOGGER.info("【Apollo-config-change】>> key={} , propertyName={} , oldValue={} , newValue={} ",
key, change.getPropertyName(), change.getOldValue(), change.getNewValue());
//是否為日志配置
if (StringUtils.containsIgnoreCase(key, LOGGER_TAG)) {
//日志配置刷新
changeLoggingLevel(key, change);
continue;
}
// 更新相應(yīng)的bean的屬性值,主要是存在@ConfigurationProperties注解的bean
this.applicationContext.publishEvent(new EnvironmentChangeEvent(changeEvent.changedKeys()));
}
LOGGER.info("【Apollo-config-change】>> end");
}
/**
* 刷新日志級(jí)別
*/
private void changeLoggingLevel(String key, ConfigChange change) {
if (null == loggingSystem) {
return;
}
String newLevel = change.getNewValue();
LogLevel level = LogLevel.valueOf(newLevel.toUpperCase());
loggingSystem.setLogLevel(key.replace(LOGGER_TAG, ""), level);
LOGGER.info("【Apollo-logger-config-change】>> {} -> {}", key, newLevel);
}
}
4.4 其它問(wèn)題
4.4.1配置文件與配置中心同時(shí)存在配置,啟用的是那一份
apollo 配置開(kāi)關(guān)開(kāi)啟情況下,配置中心配置會(huì)覆蓋本地配置
注:配置開(kāi)關(guān) apollo.bootstrap.enabled=true
4.4.2 配置中心掛掉會(huì)影響已發(fā)布的項(xiàng)目嗎?
項(xiàng)目啟動(dòng)后配置會(huì)存在緩存中,配置中心掛掉,已發(fā)布的項(xiàng)目不影響
4.4.3 是否支持更新端口配置
支持更新端口配置,但是必需要重啟生效,同時(shí)也需要考慮服務(wù)器的端口占用問(wèn)題。
附錄
Apollo官方文檔相關(guān)
官方源碼地址:https://github.com/ctripcorp/apollo
官方演示環(huán)境(Demo):
106.54.227.205
賬號(hào)/密碼:apollo/admin
快速搭建本地測(cè)試環(huán)境
分布式部署指南(生產(chǎn)環(huán)境建議使用)
到此這篇關(guān)于SpringBoot整合Apollo配置中心快速使用詳解的文章就介紹到這了,更多相關(guān)SpringBoot整合Apollo配置內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java后臺(tái)接口開(kāi)發(fā)初步實(shí)戰(zhàn)教程
下面小編就為大家分享一篇 Java后臺(tái)接口開(kāi)發(fā)初步實(shí)戰(zhàn)教程,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-01-01
關(guān)于@EnableGlobalMethodSecurity注解的用法解讀
這篇文章主要介紹了關(guān)于@EnableGlobalMethodSecurity注解的用法解讀,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-03-03
Java報(bào)錯(cuò)org.hibernate.TypeMismatchException的解決方法
在Java開(kāi)發(fā)領(lǐng)域,尤其是涉及到數(shù)據(jù)持久化的項(xiàng)目中,Hibernate是一款廣泛使用的強(qiáng)大工具,然而,可能會(huì)在使用過(guò)程中遭遇各種報(bào)錯(cuò),其中org.hibernate.TypeMismatchException就是一個(gè)讓人頭疼的問(wèn)題,下面我們一起深入剖析這個(gè)報(bào)錯(cuò)信息2024-11-11
Maven中plugins與pluginManagement的區(qū)別說(shuō)明
這篇文章主要介紹了Maven中plugins與pluginManagement的區(qū)別說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09
JavaCV實(shí)現(xiàn)多個(gè)MP4視頻的合并
這篇文章主要為大家詳細(xì)介紹了如何使用javacv和ffmpeg框架實(shí)現(xiàn)簡(jiǎn)單快速的合并mp4文件的視頻和音頻,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-10-10

