spring老项目升级spring-boot之dubbo升级

spring老项目升级spring-boot之dubbo升级

巨大的建筑,总是由一木一石叠起来的,我们何妨做做这一木一石呢?我时常做些零碎事,就是为此。

这是对的,但是我没有说过这句话! —— 鲁迅

问题的开始

之前老的spring项目使用dubbo的时候,都是使用的xml的方式。这篇文章主要是站在consumer端的角度出发,也就是 provider不变的情况下(仍然是xml),怎样先升级consumer端,来实现项目的正常运行。

我相信,使用xml的老项目的配置文件一般长这样.

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"
       xmlns="http://www.springframework.org/schema/beans"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
        http://code.alibabatech.com/schema/dubbo
        http://code.alibabatech.com/schema/dubbo/dubbo.xsd">

    <dubbo:application name="api-consumer" owner="ennz"/>
    <dubbo:registry group="${dubbo.group}" check="false" protocol="zookeeper"
                    address="${zookeeper.hosts}"/>
    
    <dubbo:reference version="0.0.1" check="false" id="service1"
                     interface="com.tms.bl.service.Service1"/>

    <dubbo:reference version="0.0.1" check="false" id="service2"
                     interface="com.tms.bl.service.Service2"/>
</beans>

Application中引入这个文件,会出现报错

@SpringBootApplication
@ImportResource(locations = {"classpath:dubbo_consumer.xml"})
public class AdminApiApplication  {
  private static final Logger logger = LoggerFactory.getLogger(AdminApiApplication.class);
  public static void main(String[] args) {
      SpringApplication.run(AdminApiApplication.class, args);
  }
}

报错如下:

2023-03-30 16:52:14.323 [main-SendThread()] WARN  org.apache.zookeeper.ClientCnxn - Session 0x0 for server ${zookeeper.hosts}:9090, unexpected error, closing socket connection and attempting reconnect
java.lang.IllegalArgumentException: named capturing group is missing trailing '}'
    at java.util.regex.Matcher.appendReplacement(Matcher.java:841)
    at java.util.regex.Matcher.replaceAll(Matcher.java:955)
    at java.lang.String.replaceAll(String.java:2223)
    at org.apache.zookeeper.ClientCnxn$SendThread.startConnect(ClientCnxn.java:997)
    at org.apache.zookeeper.ClientCnxn$SendThread.run(ClientCnxn.java:1064)

也就是说,相应的 ${zookeeper.hosts} 类似于这样的值,无法注入.

针对xml${zookeeper.hosts}无法注入的思考

无法注入,是否可以通过使用 @Configuration注解的方式,来注入相应的值

新增配置类,但是不完全取代xml

在xml中,有两个标签如下:

 <!-- 对应java类是 ApplicationConfig -->
<dubbo:application name="api-consumer" owner="ennz"/>
 <!-- 对应java类是 RegistryConfig -->
<dubbo:registry group="${dubbo.group}" check="false" protocol="zookeeper"
                    address="${zookeeper.hosts}"/>

把这两个用java配置类来实现,至于有很多已经写好的<dubbo:reference />则继续使用xml的方式,通过@ImportResource来实现。java配置类如下:

@Configuration
@ImportResource("classpath:dubbo-consumer.xml")
public class DubboAdasConsumerConfig {
  private Logger logger = LoggerFactory.getLogger(DubboAdasConsumerConfig.class);

  @Value("${dubbo.zookeepers}")
  private String dubboZookeepers;

  @Value("${dubbo.group}")
  private String dubboGroup;

  /*相当于consumer.xml中的:<dubbo:application name="consumer"/>*/
  @Bean
  public ApplicationConfig gpsApplicationConfig() {
    ApplicationConfig applicationConfig = new ApplicationConfig();
    applicationConfig.setName("api-consumer");
    applicationConfig.setOwner("ennz");
    return applicationConfig;
  }

  /*相当于:<dubbo:registry address="39.108.125.227:2181" protocol="zookeeper"/>*/
  @Bean
  public RegistryConfig adasRegistryConfig() {
    RegistryConfig registryConfig = new RegistryConfig();
    registryConfig.setProtocol("zookeeper");
    registryConfig.setCheck(false);
    registryConfig.setAddress(dubboZookeepers);
    registryConfig.setGroup(dubboGroup);
    logger.info("adasRegistryConfig:{}", registryConfig.getAddress());
    return registryConfig;
  }
}

其中 dubbo-consumer.xml 中,去掉这两个类的配置,只剩下主要service的注入:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:dubbo="http://code.alibabatech.com/schema/dubbo"
       xmlns="http://www.springframework.org/schema/beans"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
        http://code.alibabatech.com/schema/dubbo
        http://code.alibabatech.com/schema/dubbo/dubbo.xsd">
    
    <dubbo:reference version="0.0.1" check="false" id="service1"
                     interface="com.tms.bl.service.Service1"/>

    <dubbo:reference version="0.0.1" check="false" id="service2"
                     interface="com.tms.bl.service.Service2"/>
</beans>

但是运行之后,仍然报错,是dubbo中注入的service为空。我这里是生成shiro的时候,用到了某个dubbo服务。报错:register dubbo:null

Caused by: java.lang.IllegalStateException: registry address == null

然后,输出了一下 registryConfig.getAddress()的值,是null,也就是@Value("${dubbo.zookeepers}")失效,并没有注入成功.

分析一下原因,应该跟加载顺序有关.

  1. 先加载bean
  2. 加载配置类,加载@Value值。
  3. 执行第一步的时候,发现有的是dubbo的,但是想要加载dubbo的,就需要先找到dubbo的配置,就先加载了dubbo的配置类.但是,此时@Value还未生效,是null.

使用最基础的方法,读取文件,提前找到配置。

配置文件一般都提取出来,放到config文件夹下面.

直接读取它,获取属性配置。

@SpringBootApplication
@ImportResource(locations = {"classpath:applicationContext.xml"})
public class AdminApiApplication  {
  private static final Logger logger = LoggerFactory.getLogger(AdminApiApplication.class);

  public static void main(String[] args) {
    try {
      setConfigProperties();
      SpringApplication.run(AdminApiApplication.class, args);
      System.out.println("print start");
    } catch (Exception e) {
      e.printStackTrace();
    }
  }

  public static void setConfigProperties() {
    Properties config = new Properties();
    try {
      File file = new File("/data/config/config.properties");
      FileInputStream fileInputStream = new FileInputStream(file);
      InputStream is = fileInputStream;
      config.load(is);
    } catch (IOException var5) {
      throw new RuntimeException("An error occurred while reaed exceptions", var5);
    }
    CacheManager.configProperties = config;
    logger.info("config:{}", JSON.toJSONString(config));
  }

}

其中 CacheManager.configProperties是一个静态变量,简单的存储器.

public class CacheManager {
  /**
   * config中的变量.
   */
  public static Properties configProperties;
}

使用注入@Value的地方,不使用@Value来注入,直接取值

  @Bean
  public RegistryConfig gpsRegistryConfig() {
    RegistryConfig registryConfig = new RegistryConfig();
    registryConfig.setProtocol("zookeeper");
    registryConfig.setCheck(false);
    registryConfig.setAddress(CacheManager.configProperties.getProperty("dubbo.zookeepers"));
    registryConfig.setGroup(CacheManager.configProperties.getProperty("dubbo.group"));
    return registryConfig;
  }

运行,可以启动成功.

有没有更好的方法

使用 EnvironmentPostProcessor,这里会在加载配置类之前执行。增加类,实现AdminApiApplication中的setConfigProperties方法.

/**
 *
 * @author liuzhenning
 * @version 0.0.1
 * @since 0.0.1 2022-11-25
 */
public class CustomerConfigLoadProcessor implements EnvironmentPostProcessor {

  private Logger logger = LoggerFactory.getLogger(CustomerConfigLoadProcessor.class);

  @Override
  public void postProcessEnvironment(ConfigurableEnvironment environment, SpringApplication application) {
    System.out.println("postProcessEnvironment begin=====================================");
    Properties config = new Properties();

    this.addProperties(config, "/data/config/config.properties");
    this.addProperties(config, "E://temp/config/config.properties");

    CacheManager.configProperties = config;
    System.out.println("postProcessEnvironment done");
  }

  private void addProperties(Properties config, String fileName) {
    try {
      Properties properties = new Properties();
      File file = new File(fileName);
      if (!file.exists()) {
        return;
      }
      InputStream is = new FileInputStream(file);
      properties.load(is);
      config.putAll(properties);
      is.close();
    } catch (IOException var5) {
      logger.error("error", var5);
    }
  }

}

增加该类后,spring并不会加载它,还需要配置一下

resources下面增加文件夹META-INF/spring.factories,增加配置

org.springframework.boot.env.EnvironmentPostProcessor=\
  com.xxxx.CustomerConfigLoadProcessor

去掉AdminApiApplication 中的setConfigProperties(),运行,成功启动.

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,835评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,598评论 1 295
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,569评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,159评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,533评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,710评论 1 222
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,923评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,674评论 0 203
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,421评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,622评论 2 245
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,115评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,428评论 2 254
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,114评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,097评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,875评论 0 197
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,753评论 2 276
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,649评论 2 271

推荐阅读更多精彩内容