问题描述

近期开发项目,将Dubbo的配置全部外部化到动态配置中心。这里配置中心我使用的是Apollo。

@Configuration public class DubboConfig {   @Bean   public ConfigCenterConfig configCenterConfig() {     ConfigCenterConfig configCenterConfig = new ConfigCenterConfig();     configCenterConfig.setAddress("apollo.xxxxx.com");     configCenterConfig.setProtocol("apollo");     configCenterConfig.setNamespace("dubbo");     configCenterConfig.setGroup(null);     return configCenterConfig;   } } 

这里使用dubbo这个namespace。

其余配置全部配置在这个里面。

然后遇到一个大坑:

这里项目中定义了一个Service,也就是作为Provider提供了版本为provider.auth.version的服务。

package io.github.slankka.provider;  @Service(version = "${provider.auth.version:1.0}") @Slf4j public class AuthApiService implements IAuthApi 

项目启动之后,注册到注册中心的版本是1.0。

在dubbo命名空间中定义了provider.auth.version=20000,但是启动还是1.0

问题分析

Apollo启动模式分为两种,一种是在 BeanDefinition阶段就将配置注入Spring容器内的Environment。另一种是在PostBeanFactory的时候启动。
而Dubbo启动的时候,所有的Service是ServiceBean的实例对应生成的代理类,所有的Reference是ReferenceBean的实例对应生成的代理类。
因此如果这里使用placeholder要被Spring识别,那么将必须选用第一种启动方式。

问题解决

在本地项目的application.properties中配置:

apollo.bootstrap.enabled = true apollo.bootstrap.namespaces = application,dubbo

注意,这里apollo.bootstrap.namespaces 中加入了dubbo,这里有一个疑问:
之前配置Dubbo的时候不是定义了ConfigCenterConfig的namespace了吗,为什么还要定义一次。

 configCenterConfig.setNamespace("dubbo");

原因是:
Dubbo的ServiceBean在被Spring工厂构建出来的时候,就需要这个变量。但如果没有配置dubbo到apollo.bootstrap.namespaces,Spring会报错。
如果接口上写了默认版本,上例是1.0,则Spring不会报错,但是构建出来的对应ServiceBean中的版本将会是1.0,而Dubbo启动之后依然不会修改这个版本。

新的问题

为什么Dubbo的namespace中配置了这个变量,而且Dubbo在启动早起阶段就已经拉到了这些变量,但版本仍旧没有发生改变?

看一下Dubbo代码:

//package org.apache.dubbo.common.config;  public abstract class AbstractPrefixConfiguration implements Configuration {     protected String id;     protected String prefix;      public AbstractPrefixConfiguration(String prefix, String id) {         if (StringUtils.isNotEmpty(prefix) && !prefix.endsWith(".")) {             this.prefix = prefix + ".";         } else {             this.prefix = prefix;         }         this.id = id;     }      @Override     public Object getProperty(String key, Object defaultValue) {         Object value = null;         if (StringUtils.isNo