spring老項(xiàng)目升級(jí)spring-boot之dubbo升級(jí)

spring老項(xiàng)目升級(jí)spring-boot之dubbo升級(jí)

巨大的建筑,總是由一木一石疊起來(lái)的恒水,我們何妨做做這一木一石呢?我時(shí)常做些零碎事浑度,就是為此寇窑。

這是對(duì)的,但是我沒(méi)有說(shuō)過(guò)這句話箩张! —— 魯迅

問(wèn)題的開(kāi)始

之前老的spring項(xiàng)目使用dubbo的時(shí)候甩骏,都是使用的xml的方式。這篇文章主要是站在consumer端的角度出發(fā)先慷,也就是 provider不變的情況下(仍然是xml),怎樣先升級(jí)consumer端饮笛,來(lái)實(shí)現(xiàn)項(xiàng)目的正常運(yùn)行。

我相信论熙,使用xml的老項(xiàng)目的配置文件一般長(zhǎng)這樣.

<?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中引入這個(gè)文件福青,會(huì)出現(xiàn)報(bào)錯(cuò)

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

報(bào)錯(cuò)如下:

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)

也就是說(shuō),相應(yīng)的 ${zookeeper.hosts} 類似于這樣的值脓诡,無(wú)法注入.

針對(duì)xml${zookeeper.hosts}無(wú)法注入的思考

無(wú)法注入无午,是否可以通過(guò)使用 @Configuration注解的方式,來(lái)注入相應(yīng)的值

新增配置類祝谚,但是不完全取代xml

在xml中宪迟,有兩個(gè)標(biāo)簽如下:

 <!-- 對(duì)應(yīng)java類是 ApplicationConfig -->
<dubbo:application name="api-consumer" owner="ennz"/>
 <!-- 對(duì)應(yīng)java類是 RegistryConfig -->
<dubbo:registry group="${dubbo.group}" check="false" protocol="zookeeper"
                    address="${zookeeper.hosts}"/>

把這兩個(gè)用java配置類來(lái)實(shí)現(xiàn),至于有很多已經(jīng)寫(xiě)好的<dubbo:reference />則繼續(xù)使用xml的方式交惯,通過(guò)@ImportResource來(lái)實(shí)現(xiàn)次泽。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;

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

  /*相當(dāng)于:<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 中穿仪,去掉這兩個(gè)類的配置,只剩下主要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>

但是運(yùn)行之后意荤,仍然報(bào)錯(cuò)啊片,是dubbo中注入的service為空。我這里是生成shiro的時(shí)候玖像,用到了某個(gè)dubbo服務(wù)紫谷。報(bào)錯(cuò):register dubbo:null

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

然后,輸出了一下 registryConfig.getAddress()的值御铃,是null,也就是@Value("${dubbo.zookeepers}")失效碴里,并沒(méi)有注入成功.

分析一下原因,應(yīng)該跟加載順序有關(guān).

  1. 先加載bean
  2. 加載配置類上真,加載@Value值咬腋。
  3. 執(zhí)行第一步的時(shí)候,發(fā)現(xiàn)有的是dubbo的睡互,但是想要加載dubbo的根竿,就需要先找到dubbo的配置,就先加載了dubbo的配置類.但是就珠,此時(shí)@Value還未生效寇壳,是null.

使用最基礎(chǔ)的方法,讀取文件妻怎,提前找到配置壳炎。

配置文件一般都提取出來(lái),放到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是一個(gè)靜態(tài)變量,簡(jiǎn)單的存儲(chǔ)器.

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

使用注入@Value的地方榛丢,不使用@Value來(lái)注入铲球,直接取值

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

運(yùn)行,可以啟動(dòng)成功.

有沒(méi)有更好的方法

使用 EnvironmentPostProcessor晰赞,這里會(huì)在加載配置類之前執(zhí)行稼病。增加類,實(shí)現(xiàn)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并不會(huì)加載它然走,還需要配置一下

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

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

去掉AdminApiApplication 中的setConfigProperties(),運(yùn)行,成功啟動(dòng).

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末戏挡,一起剝皮案震驚了整個(gè)濱河市丰刊,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌增拥,老刑警劉巖啄巧,帶你破解...
    沈念sama閱讀 218,451評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異掌栅,居然都是意外死亡秩仆,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)猾封,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)澄耍,“玉大人,你說(shuō)我怎么就攤上這事晌缘∑肓” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,782評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵磷箕,是天一觀的道長(zhǎng)选酗。 經(jīng)常有香客問(wèn)我,道長(zhǎng)岳枷,這世上最難降的妖魔是什么芒填? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,709評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮空繁,結(jié)果婚禮上殿衰,老公的妹妹穿的比我還像新娘。我一直安慰自己盛泡,他們只是感情好闷祥,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,733評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著傲诵,像睡著了一般凯砍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上掰吕,一...
    開(kāi)封第一講書(shū)人閱讀 51,578評(píng)論 1 305
  • 那天果覆,我揣著相機(jī)與錄音,去河邊找鬼殖熟。 笑死局待,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的菱属。 我是一名探鬼主播钳榨,決...
    沈念sama閱讀 40,320評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼纽门!你這毒婦竟也來(lái)了薛耻?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,241評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤赏陵,失蹤者是張志新(化名)和其女友劉穎饼齿,沒(méi)想到半個(gè)月后饲漾,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,686評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡缕溉,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,878評(píng)論 3 336
  • 正文 我和宋清朗相戀三年考传,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片证鸥。...
    茶點(diǎn)故事閱讀 39,992評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡僚楞,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出枉层,到底是詐尸還是另有隱情泉褐,我是刑警寧澤,帶...
    沈念sama閱讀 35,715評(píng)論 5 346
  • 正文 年R本政府宣布鸟蜡,位于F島的核電站膜赃,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏矩欠。R本人自食惡果不足惜财剖,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,336評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望癌淮。 院中可真熱鬧躺坟,春花似錦、人聲如沸乳蓄。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,912評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)虚倒。三九已至美侦,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間魂奥,已是汗流浹背菠剩。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,040評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留耻煤,地道東北人具壮。 一個(gè)月前我還...
    沈念sama閱讀 48,173評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像哈蝇,于是被迫代替她去往敵國(guó)和親棺妓。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,947評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容