Spring Boot 应用打 WAR 包后无法注册到 Nacos怎么办

你好,我是柳岸花开。

在微服务架构中,服务注册与发现是至关重要的一环。Nacos 作为阿里巴巴开源的注册中心,能够很好地满足这一需求。然而,在将 Spring Boot 应用打包成 WAR 部署到外部服务器时,可能会遇到服务无法注册到 Nacos 的问题。本文将详细讲解这一问题的解决方案。

问题描述

在开发过程中,通常使用 JAR 包运行 Spring Boot 应用,这种方式下,服务注册到 Nacos 通常没有问题。然而,当我们将 Spring Boot 应用打包成 WAR 并部署到外部 Tomcat 服务器时,可能会遇到服务无法注册到 Nacos 的情况。其原因主要是应用获取不到正确的服务器端口。

解决方案

为了在 WAR 包部署时正确地注册服务到 Nacos,我们需要动态地获取实际使用的服务器端口,并将其设置到 Nacos 的服务注册中。以下是一个具体的实现代码:

复制代码
import com.alibaba.cloud.nacos.registry.NacosAutoServiceRegistration;
import lombok.extern.slf4j.Slf4j;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.stereotype.Component;

import javax.management.MBeanServer;
import javax.management.ObjectName;
import javax.management.Query;
import java.lang.management.ManagementFactory;
import java.util.Set;

/**
 * 解决 Spring Boot 应用打 WAR 包后无法注册到 Nacos 的问题
 * 通过动态获取服务器端口并注册到 Nacos
 */
@Slf4j
@ConditionalOnProperty(prefix = "project.deploy", name = "mode", havingValue = "war")
@Component
public class NacosConfig implements ApplicationRunner {

    @Autowired
    private NacosAutoServiceRegistration registration;

    @Value("${server.port:8080}")
    Integer port;

    @Override
    public void run(ApplicationArguments args) {
        if (registration != null && port != null) {
            Integer serverPort = port;
            try {
                serverPort = new Integer(getServerPort());
            } catch (Exception e) {
                log.warn("getServerPort warn", e);
                log.info("getServerPort fail, use the config-file's port {}", port);
            }
            registration.setPort(serverPort);
            registration.start();
        }
    }

    /**
     * 获取实际使用的服务器端口
     */
    public String getServerPort() throws Exception {
        MBeanServer beanServer = ManagementFactory.getPlatformMBeanServer();
        Set<ObjectName> objectNames = beanServer.queryNames(new ObjectName("*:type=Connector,*"), Query.match(Query.attr("protocol"), Query.value("HTTP/1.1")));
        String port = objectNames.iterator().next().getKeyProperty("port");
        log.debug("getServerPort {}", port);
        return port;
    }
}

代码解析

1. 类与注解

复制代码
@Slf4j
@ConditionalOnProperty(prefix = "project.deploy", name = "mode", havingValue = "war")
@Component
public class NacosConfig implements ApplicationRunner {
  • @Slf4j:Lombok 提供的注解,用于生成日志记录器。
  • @ConditionalOnProperty:只有在 project.deploy.mode 属性值为 war 时,才会创建这个 NacosConfig bean。
  • @Component:标记为 Spring 的组件,使其能够被 Spring 扫描和管理。
  • ApplicationRunner:实现该接口的 run 方法将在应用启动时运行。

2. 自动注入与配置

复制代码
@Autowired
private NacosAutoServiceRegistration registration;

@Value("${server.port:8080}")
Integer port;
  • @Autowired:注入 Nacos 的服务注册类 NacosAutoServiceRegistration
  • @Value:注入配置文件中的端口号,如果未配置则默认使用 6888 端口。

3. 启动时设置端口

复制代码
@Override
public void run(ApplicationArguments args) {
    if (registration != null && port != null) {
        Integer serverPort = port;
        try {
            serverPort = new Integer(getServerPort());
        } catch (Exception e) {
            log.warn("getServerPort warn", e);
            log.info("getServerPort fail, use the config-file's port {}", port);
        }
        registration.setPort(serverPort);
        registration.start();
    }
}
  • 在应用启动时尝试获取实际使用的服务器端口,如果获取失败则使用配置文件中的端口。
  • 将端口设置到 Nacos 的服务注册中,并启动服务注册。

4. 获取实际端口

复制代码
public String getServerPort() throws Exception {
    MBeanServer beanServer = ManagementFactory.getPlatformMBeanServer();
    Set<ObjectName> objectNames = beanServer.queryNames(new ObjectName("*:type=Connector,*"), Query.match(Query.attr("protocol"), Query.value("HTTP/1.1")));
    String port = objectNames.iterator().next().getKeyProperty("port");
    log.debug("getServerPort {}", port);
    return port;
}
  • 通过 JMX 管理接口获取实际使用的 HTTP 端口。

总结

通过上述代码,我们可以实现在 Spring Boot 应用启动时,动态获取并设置 Nacos 注册的端口号。这对于在不同环境中部署应用非常有用,可以避免硬编码端口号带来的问题。同时,结合 Nacos 的服务注册与发现功能,可以更加灵活地管理微服务架构中的各个服务。

希望本文能对大家有所帮助,如果有任何疑问或建议,欢迎在下方留言交流。

👇关注我,下期了解👇

Spring配置类源码深度解析

回复 222,获取Java面试题合集

关于我

一枚爱折腾的Java程序猿,专注Spring干货。把路上的问题记录下来,帮助那些和我一样的人。

好奇心强,喜欢并深入研究古天文。

崇尚 个人系统创建,做一些时间越长越有价值的事情。思考 把时间留下来 又 每刻都是新的。

本文由mdnice多平台发布

相关推荐
计算机徐师兄几秒前
Java基于微信小程序的物流管理系统【附源码、文档说明】
java·微信小程序·物流管理系统·java物流管理系统小程序·物流管理系统小程序·物流管理系统微信小程序·java物流管理系统微信小程序
青云交9 分钟前
Java 大视界 -- Java 大数据机器学习模型在金融风险管理体系构建与风险防范能力提升中的应用(435)
java·大数据·机器学习·spark·模型可解释性·金融风控·实时风控
跟‘码’死磕14 分钟前
springboot集成钉钉群内发送消息
java·spring boot·钉钉
0和1的舞者15 分钟前
SpringBoot配置文件
java·spring boot·后端·web·配置·spirng
cike_y18 分钟前
JavaWeb之过滤器Filter&监听器
java·servlet·javaweb
多则惑少则明18 分钟前
SpringAI框架接入-jdk升级21后报错“run failed: Unsupported class file major version 65”
java·后端·spring·springai
uup22 分钟前
线程池中任务堆积与饥饿死锁问题
java
deng-c-f26 分钟前
C/C++内置库函数(2):智能指针
java·c语言·c++
毕设源码-朱学姐27 分钟前
【开题答辩全过程】以 基于SSM框架的餐厅点餐系统的设计与实现为例,包含答辩的问题和答案
java·eclipse
pursue.dreams27 分钟前
Java实现企业微信机器人消息推送:文本消息与文件推送完整指南
java·机器人·企业微信