详解Springboot工程中如何快速判断web应用服务器类型

 更新时间:2026年04月13日 09:31:38   作者:爱码少年 00fly.online  
在 Spring Boot 工程中快速判断 Web 应用服务器类型,可以从运行时环境、Spring Boot 特有 API、底层类加载器、HTTP 响应和外部命令等多个层面入手,下面小编就和大家简单介绍一下吧

在 Spring Boot 工程中快速判断 Web 应用服务器类型,可以从运行时环境、Spring Boot 特有 API、底层类加载器、HTTP 响应和外部命令等多个层面入手。下面是几种常见方法的速览表:

方法原理简述代码复杂度通用性推荐度
注入 ServletContext直接调用 getServerInfo() 获取服务器信息★☆☆☆☆极高 (适用于所有 Servlet 容器)⭐⭐⭐⭐⭐
检查特定类是否存在通过 Class.forName() 加载特定服务器的关键类★★☆☆☆⭐⭐⭐⭐
利用 Spring Boot 条件注解使用 @ConditionalOnClass 注解声明 Bean★★☆☆☆一般 (适用于 Spring 环境)⭐⭐⭐
监听容器初始化事件在应用启动时监听 ServletWebServerInitializedEvent 事件★★☆☆☆一般 (适用于 Spring Boot)⭐⭐⭐
查看启动日志直接观察应用启动时的控制台日志输出☆☆☆☆☆极高⭐⭐⭐⭐⭐
使用 Actuator 端点访问 /actuator/env 端点查看服务器相关配置★★☆☆☆一般⭐⭐⭐
检查特定系统属性或 JMX读取 java.vm.vendor 或查询 JMX MBean★★★☆☆⭐⭐
通过外部命令或端口号推断使用 netstat 或 ps 命令在操作系统层面查询★★★☆☆低 (仅限手动排查)

方法一:通过 ServletContext 获取服务器信息(推荐)

这是最直接、最可靠的方式。你可以在任何 Spring Bean 中注入 ServletContext 对象,并调用 getServerInfo() 方法。该方法会返回一个字符串,其中包含了当前正在运行的 Servlet 容器的名称和版本信息

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import javax.servlet.ServletContext;
@Service
public class ServerInfoService {
    @Autowired
    private ServletContext servletContext;
    public void printServerInfo() {
        String serverInfo = servletContext.getServerInfo();
        System.out.println("服务器信息: " + serverInfo);
        // 对于 Tomcat,输出可能是 "Apache Tomcat/9.0.83" 
        // 对于 Jetty,输出可能是 "Jetty/9.4.53.v20231009"
        // 对于 Undertow,输出可能是 "Undertow - 2.2.24.Final"
    }
}

根据 getServerInfo() 返回的内容进行判断时,注意对版本号部分进行清理,避免因版本差异导致匹配失败。

方法二:检查特定类的存在性

通过 Class.forName() 方法检测特定服务器的关键类是否存在于当前应用的 Classpath 中,从而判断其类型。

import org.springframework.stereotype.Component;
import javax.annotation.PostConstruct;
@Component
public class ServerDetector {
    public enum ServerType {
        TOMCAT, JETTY, UNDERTOW, JBOSS, WEBLOGIC, WEBSPHERE, UNKNOWN
    }
    private static ServerType detectedServer;
    @PostConstruct
    public void detect() {
        detectedServer = determineServerType();
        System.out.println("检测到服务器类型: " + detectedServer);
    }
    public static ServerType determineServerType() {
        if (isClassPresent("org.apache.catalina.startup.Tomcat")) return ServerType.TOMCAT;
        if (isClassPresent("org.eclipse.jetty.server.Server")) return ServerType.JETTY;
        if (isClassPresent("io.undertow.Undertow")) return ServerType.UNDERTOW;
        if (isClassPresent("org.jboss.Main")) return ServerType.JBOSS;
        if (isClassPresent("weblogic.Server")) return ServerType.WEBLOGIC;
        if (isClassPresent("com.ibm.websphere.product.VersionInfo")) return ServerType.WEBSPHERE;
        return ServerType.UNKNOWN;
    }
    private static boolean isClassPresent(String className) {
        try {
            Class.forName(className);
            return true;
        } catch (ClassNotFoundException e) {
            return false;
        }
    }
}

方法三:利用 Spring Boot 的条件注解

这种方式适合用于控制某些特定服务器下 Bean 的加载,是一种声明式的判断方法。

import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class ServerSpecificConfig {
    @Bean
    @ConditionalOnClass(name = "org.apache.catalina.startup.Tomcat")
    public String tomcatOnlyBean() {
        return "仅当应用运行在 Tomcat 服务器上时,此 Bean 才会被创建";
    }
}

这个例子也可以作为快速判断的依据,当你看到 tomcatOnlyBean 在应用上下文中被创建,就可以认为当前运行的是 Tomcat 容器。

方法四:监听容器初始化事件

通过实现 ApplicationListener<ServletWebServerInitializedEvent> 接口,可以在 Servlet 容器完成初始化后获取 WebServer 实例,从而获取服务器的具体实现类,反推出其类型。

import org.springframework.boot.web.context.WebServerInitializedEvent;
import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;
@Component
public class WebServerListener implements ApplicationListener<WebServerInitializedEvent> {
    @Override
    public void onApplicationEvent(WebServerInitializedEvent event) {
        // 获取 WebServer 实例,其实现类通常是 TomcatWebServer、JettyWebServer、UndertowWebServer 等
        Class<?> webServerClass = event.getWebServer().getClass();
        System.out.println("WebServer 实现类: " + webServerClass.getName());
        // 可根据实现类名判断服务器类型
        if (webServerClass.getName().contains("Tomcat")) {
            System.out.println("当前运行在 Tomcat 服务器上");
        }
    }
}

补充说明

通用性与准确性ServletContext.getServerInfo() 是标准 Java Servlet 规范的一部分,在所有支持 Servlet 规范的容器中均可使用,且直接提供官方定义的服务器名称,是最可靠的方法。

为什么可能有“多种判断”:在某些特殊场景下(例如需要与不支持 Servlet API 的老旧代码集成),可能需要采用其他方式。另外,getServerInfo() 返回的字符串版本号格式因服务器而异,有时需要自行解析。

方法选择建议:对绝大多数 Spring Boot 应用,最推荐 ServletContext.getServerInfo()。若无法获取 ServletContext,可使用“检查类是否存在”的方法作为替代。若要根据服务器类型选择性加载 Bean,应优先使用 Spring 的条件注解。

方法补充

该代码片段通过检查Spring应用上下文中的Bean定义来判断当前使用的是Tomcat还是Jetty容器。使用@PostConstruct注解在初始化时执行检测逻辑,通过扫描Bean名称是否包含EmbeddedTomcat或EmbeddedJetty来设置相应的布尔标志,并输出日志记录检测结果。这种动态识别方式适用于需要针对不同嵌入式容器做差异化处理的场景。

核心源码

    boolean isTomcat;
    
    boolean isJetty;
    
    @Autowired
    ApplicationContext applicationContext;
    
    @PostConstruct
    private void init()
    {
        isTomcat = Stream.of(applicationContext.getBeanDefinitionNames()).anyMatch(name -> StringUtils.containsIgnoreCase(name, "EmbeddedTomcat"));
        isJetty = Stream.of(applicationContext.getBeanDefinitionNames()).anyMatch(name -> StringUtils.containsIgnoreCase(name, "EmbeddedJetty"));
        log.info("#### isTomcat: {}", isTomcat);
        log.info("#### isJetty: {}", isJetty);
    }

典型应用

import java.io.File;
import java.io.IOException;
import java.util.stream.Stream;

import javax.annotation.PostConstruct;

import org.apache.commons.io.FileUtils;
import org.apache.commons.lang3.RandomUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;

import com.fly.demo.common.JsonResult;

import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import lombok.extern.slf4j.Slf4j;

@Slf4j
@Api(tags = "文件上传(simple)")
@RestController
@RequestMapping("/simple/file")
public class SimpleFileController
{
    boolean isTomcat;
    
    boolean isJetty;
    
    @Autowired
    ApplicationContext applicationContext;
    
    @PostConstruct
    private void init()
    {
        isTomcat = Stream.of(applicationContext.getBeanDefinitionNames()).anyMatch(name -> StringUtils.containsIgnoreCase(name, "EmbeddedTomcat"));
        isJetty = Stream.of(applicationContext.getBeanDefinitionNames()).anyMatch(name -> StringUtils.containsIgnoreCase(name, "EmbeddedJetty"));
        log.info("#### isTomcat: {}", isTomcat);
        log.info("#### isJetty: {}", isJetty);
    }
    
    @ApiOperation("文件上传")
    @PostMapping("/upload")
    public JsonResult<?> upload(@RequestParam MultipartFile file)
        throws IOException
    {
        File rootDir = new File("upload");
        File dest;
        if (isTomcat)
        {
            if (RandomUtils.nextBoolean())
            {
                log.info("### transferTo");
                dest = new File(rootDir.getCanonicalPath() + File.separator + file.getOriginalFilename());
                file.transferTo(dest);
            }
            else
            {
                log.info("### copyInputStreamToFile");
                dest = new File(rootDir, file.getOriginalFilename());
                FileUtils.copyInputStreamToFile(file.getInputStream(), dest);
            }
        }
        else
        {
            log.info("### copyInputStreamToFile");
            dest = new File(rootDir, file.getOriginalFilename());
            FileUtils.copyInputStreamToFile(file.getInputStream(), dest);
        } 
        return JsonResult.success(dest.getName());
    }
}

到此这篇关于详解Springboot工程中如何快速判断web应用服务器类型的文章就介绍到这了,更多相关Springboot判断web应用服务器类型内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java经典排序算法之插入排序代码实例

    Java经典排序算法之插入排序代码实例

    这篇文章主要介绍了Java经典排序算法之插入排序代码实例,插入排序是一种最简单直观的排序算法,它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入,需要的朋友可以参考下
    2023-10-10
  • 如何通过ServletInputStream读取http请求传入的数据

    如何通过ServletInputStream读取http请求传入的数据

    这篇文章主要介绍了如何通过ServletInputStream读取http请求传入的数据,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • JPA原生SQL实现增删改查的示例详解

    JPA原生SQL实现增删改查的示例详解

    JPA除了对JPQL提供支持外,还对原生SQL语句也提供了支持。本文将利用生SQL实现增删改查功能,文中的示例代码讲解详细,需要的可以参考一下
    2022-09-09
  • java中的4种循环方法示例详情

    java中的4种循环方法示例详情

    大家好,本篇文章主要讲的是java中的4种循环方法示例详情,感兴趣的同学赶快来看一看吧,对你有帮助的话记得收藏一下,方便下次浏览
    2021-12-12
  • 一文吃透消息队列RocketMQ实现消费幂等原理

    一文吃透消息队列RocketMQ实现消费幂等原理

    这篇文章主要介绍了消息队列RocketMQ实现消费幂等的全面讲解,帮助大家吃透RocketMQ消息队列消费幂等,更好的的应用与工作实践,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2024-01-01
  • 新手Hadoop安装 环境搭建

    新手Hadoop安装 环境搭建

    这篇文章主要介绍了Hadoop的安装与环境搭建教程图解,本文图文并茂给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下,希望能给您带来帮助
    2021-06-06
  • Java实现FTP批量大文件上传下载篇2

    Java实现FTP批量大文件上传下载篇2

    这篇文章主要为大家详细介绍了Java实现FTP批量大文件上传下载的强化篇,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-08-08
  • Java实现ArrayList自动扩容

    Java实现ArrayList自动扩容

    ArrayList的扩容规则是非常简单的,它会根据需要自动扩容,本文就来介绍一下Java实现ArrayList自动扩容,具有一定的参考价值,感兴趣的可以了解一下
    2023-12-12
  • Java面向对象特性深入刨析封装

    Java面向对象特性深入刨析封装

    封装是一个非常广泛的概念,小到一个属性的封装,大到一个框架或者一个项目的封装,下面这篇文章主要给大家介绍了关于java中封装的那点事,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-05-05
  • Maven 安装目录的详细介绍

    Maven 安装目录的详细介绍

    这篇文章主要介绍了Maven 安装目录的详细介绍的相关资料,这里对Maven进行了解读,希望通过本文可以帮助到大家,需要的朋友可以参考下
    2017-08-08

最新评论