SpringBoot整合Apollo配置中心快速使用详解

作者:程序员小强 时间:2022-12-25 17:00:34 

一、简介

1.Apollo 是什么?Apollo(阿波罗)是携程框架部门研发的分布式配置中心。服务端基于Spring Boot和Spring Cloud开发。

2.为什么要使用Apollo?

  • 安全性:配置跟随源代码保存在代码库中,容易造成配置泄漏

  • 时效性:普通方式配置,修改配置,需要重启服务才能生效

  • 局限性:无法支持动态调整:例如日志开关、功能开关

二、使用

1. 测试项目搭建

注:本文主要介绍SpringBoot 整合 Apollo 实现动态配置

1.1 添加Maven依赖


<dependency>
 <groupId>com.ctrip.framework.apollo</groupId>
 <artifactId>apollo-client</artifactId>
 <version>1.3.0</version>
</dependency>

1.2 配置文件


# apollo集成
# apollo 配置应用的 appid
app.id=springboot-apollo-demo1
# apollo meta-server地址,一般同config-server地址
apollo.meta=http://192.168.0.153:8080
#启用apollo配置开关
apollo.bootstrap.enabled=true
apollo.bootstrap.eagerLoad.enabled=true
# apollo 使用配置的命名空间,多个以逗号分隔
apollo.bootstrap.namespaces = application

配置说明:

  • app.id:在配置中心配置的应用身份信息。

  • apollo.bootstrap.enabled:在应用启动阶段是否向Spring容器注入被托管的properties文件配置信息。

  • apollo.bootstrap.eagerLoad.enabled:将Apollo配置加载提到初始化日志系统之前。

  • apollo.bootstrap.namespaces:配置的命名空间,多个逗号分隔,一个namespace相当于一个配置文件。

  • **apollo.meta:**当前环境服务配置地址,生产环境建议至少双节点,可以填写多个逗号分隔,使用一个单独的域,如 http://config.xxx.com(由nginx等软件负载平衡器支持),而不是多个IP地址,因为服务器可能会扩展或缩小。

图示说明:

SpringBoot整合Apollo配置中心快速使用详解

1.3 添加启动类


@SpringBootApplication(exclude = {DataSourceAutoConfiguration.class, DataSourceTransactionManagerAutoConfiguration.class})
public class SpringbootApolloApplication {
   public static void main(String[] args) {
       SpringApplication.run(SpringbootApolloApplication.class, args);
   }
}

1.4 添加配置开关类

基于@Value注解配置


@Component
public class ValueStyleProperty {

@Value("${apollo.value.demoKey1}")
   private String demoKey1;

@Value("${apollo.value.demoKey2}")
   private String demoKey2;
   //省略 get/set 方法
}

1.5 添加测试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 创建项目

SpringBoot整合Apollo配置中心快速使用详解

2.2 填写配置信息

SpringBoot整合Apollo配置中心快速使用详解

配置说明:

  • 部门:选择应用所在的部门。(可自定义部门)

  • 应用AppId:用来标识应用身份的唯一id,格式为string,需与application.properties中配置的app.id一致。

  • 应用名称:应用名,仅用于界面展示。

  • 应用负责人:选择的人默认会成为该项目的管理员,具备项目权限管理、集群创建、Namespace创建等权限。

项目配置主页截图

SpringBoot整合Apollo配置中心快速使用详解

2.3 添加配置

2.3.1 表格形式单个添加

注:不能批量操作

SpringBoot整合Apollo配置中心快速使用详解

SpringBoot整合Apollo配置中心快速使用详解

2.3.2 文本形式批量添加

注:可实现批量操作

SpringBoot整合Apollo配置中心快速使用详解

SpringBoot整合Apollo配置中心快速使用详解

SpringBoot整合Apollo配置中心快速使用详解

2.4 发布配置

注:配置只有发布后才会生效

SpringBoot整合Apollo配置中心快速使用详解

点击发布按钮

SpringBoot整合Apollo配置中心快速使用详解

2.5 多环境同步配置

注意事项:

通过同步配置功能,可以使多个环境、集群间的配置保持一致需要注意的是,同步完之后需要发布后才会对应用生效

点击同步配置

SpringBoot整合Apollo配置中心快速使用详解

选择需要同步的配置,以及目标环境

SpringBoot整合Apollo配置中心快速使用详解

点击同步

SpringBoot整合Apollo配置中心快速使用详解

目标环境查看

SpringBoot整合Apollo配置中心快速使用详解

3. 项目启动与测试

3.1 初始启动读取测试

SpringBoot整合Apollo配置中心快速使用详解

3.2 自动更新属性测试

SpringBoot整合Apollo配置中心快速使用详解

发布后控制台变化

SpringBoot整合Apollo配置中心快速使用详解

测试输出值变化

SpringBoot整合Apollo配置中心快速使用详解

4.常见整合问题

4.1@ConfigurationProperties注解整合Apollo不生效问题

示例配置类


/**
* 公共开关,key值 属性配置
*
* @author mengqiang
*/
@Component
@ConfigurationProperties(prefix = "apollo.first.config")
public class ConfigFirstProperty {

/**
    * 测试数字
    */
   private Integer oneNumber;

/**
    * 测试字符串
    */
   private String oneStr;

/**
    * 启用标记
    */
   private Boolean oneEnableFlag;

/**
    * 税率 默认 0.03
    */
   private BigDecimal oneTaxRate = new BigDecimal("0.03");
  //省略 get/set 方法

}

解决方案
添加监听配置


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 配置监听
*/
@Configuration
public class ApolloConfigListener implements ApplicationContextAware {

/**
    * 日志
    */
   private static final Logger LOGGER = LoggerFactory.getLogger(ApolloConfigListener.class);

private ApplicationContext applicationContext;

/**
    * 配置监听
    * ApolloConfigChangeListener > value 属性默认 命名空间 "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());
       }
       // 更新相应的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日志级别未更新问题

示例配置


logging.level.com.example=info

解决方案-日志 *


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 日志-配置监听
*/
@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;

/**
    * 监听 日志配置的变化
    */
   @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");
   }

/**
    * 刷新日志级别
    */
   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日志+配置类自动刷新整合监听

注:由于 4.1与4.2监听有重合,所以最好放在一起处理


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 配置监听
*/
@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;
   }

/**
    * 配置监听
    * ApolloConfigChangeListener > value 属性默认 命名空间 "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;
           }
           // 更新相应的bean的属性值,主要是存在@ConfigurationProperties注解的bean
           this.applicationContext.publishEvent(new EnvironmentChangeEvent(changeEvent.changedKeys()));
       }
       LOGGER.info("【Apollo-config-change】>> end");
   }

/**
    * 刷新日志级别
    */
   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 其它问题

4.4.1配置文件与配置中心同时存在配置,启用的是那一份
apollo 配置开关开启情况下,配置中心配置会覆盖本地配置
注:配置开关 apollo.bootstrap.enabled=true

4.4.2 配置中心挂掉会影响已发布的项目吗?
项目启动后配置会存在缓存中,配置中心挂掉,已发布的项目不影响

4.4.3 是否支持更新端口配置
支持更新端口配置,但是必需要重启生效,同时也需要考虑服务器的端口占用问题。

附录

Apollo官方文档相关

官方源码地址:https://github.com/ctripcorp/apollo
官方演示环境(Demo):

106.54.227.205
账号/密码:apollo/admin

快速搭建本地测试环境
分布式部署指南(生产环境建议使用)

来源:https://xqiangme.blog.csdn.net/article/details/90551224

标签:SpringBoot,Apollo,配置
0
投稿

猜你喜欢

  • Mybatis-Plus的使用详解

    2022-01-18 18:39:27
  • spring boot项目打包成war在tomcat运行的全步骤

    2023-12-17 19:20:36
  • Java利用自定义注解、反射实现简单BaseDao实例

    2022-09-06 05:29:30
  • C# 利用代理爬虫网页的实现方法

    2023-02-26 18:51:43
  • 一篇文章带你了解Java Stream流

    2022-11-05 10:26:18
  • 面试官:详细谈谈Java对象的4种引用方式

    2022-09-19 04:52:20
  • 基于java.lang.IllegalArgumentException异常报错问题及解决

    2023-09-14 16:03:34
  • 如何动态修改JavaBean中注解的参数值

    2022-09-03 04:08:34
  • java 完全二叉树的构建与四种遍历方法示例

    2022-03-21 00:48:04
  • 示例解析java重载Overloading与覆盖Overriding

    2021-11-03 17:22:19
  • Flutter底部导航栏的实现方式

    2023-11-05 18:21:36
  • Java 数据结构中二叉树前中后序遍历非递归的具体实现详解

    2023-02-14 12:51:18
  • @RequestBody,@RequestParam和@Param的区别说明

    2023-07-20 06:29:03
  • SpringBoot+WebSocket实现消息推送功能

    2021-11-15 12:16:18
  • springboot aop里的@Pointcut()的配置方式

    2021-11-18 20:55:37
  • Java语言实现Blowfish加密算法完整代码分享

    2023-11-02 21:30:07
  • Android 谷歌推荐的VR实现方式(分享)

    2022-01-01 10:22:44
  • 全面分析Java文件上传

    2021-12-09 13:22:52
  • android service实现循环定时提醒功能

    2023-02-27 06:03:45
  • 详解C#中的System.Timers.Timer定时器的使用和定时自动清理内存应用

    2022-11-07 14:09:54
  • asp之家 软件编程 m.aspxhome.com