29

我将Spring Boot从 1.2.0升级到1.5.2

升级后,Tomcat 8.5在启动期间抛出FileNotFoundException

下面是其中一个异常,它抛出了超过 10 个类似的异常。

我不知道这些罐子的用途,换句话说,我没有在pom.xml<dependency>中添加这些罐子。

INFO: Starting Servlet Engine: Apache Tomcat/8.5.11
Apr 06, 2017 3:53:57 PM org.apache.tomcat.util.scan.StandardJarScanner scan
WARNING: Failed to scan [file:/C:/Users/myname/.m2/repository/com/sun/xml/ws/jaxws-rt/2.1.7/jaxws-api.jar] from classloader hierarchy
java.io.FileNotFoundException: C:\Users\myname\.m2\repository\com\sun\xml\ws\jaxws-rt\2.1.7\jaxws-api.jar (The system cannot find the file specified)
    at java.util.zip.ZipFile.open(Native Method)
    at java.util.zip.ZipFile.<init>(ZipFile.java:219)
    at java.util.zip.ZipFile.<init>(ZipFile.java:149)
    at java.util.jar.JarFile.<init>(JarFile.java:166)
    at java.util.jar.JarFile.<init>(JarFile.java:130)
    at org.apache.tomcat.util.scan.JarFileUrlJar.<init>(JarFileUrlJar.java:60)
    at org.apache.tomcat.util.scan.JarFactory.newInstance(JarFactory.java:48)
    at org.apache.tomcat.util.scan.StandardJarScanner.process(StandardJarScanner.java:338)
    at org.apache.tomcat.util.scan.StandardJarScanner.scan(StandardJarScanner.java:288)
    at org.apache.jasper.servlet.TldScanner.scanJars(TldScanner.java:262)
    at org.apache.jasper.servlet.TldScanner.scan(TldScanner.java:104)
    at org.apache.jasper.servlet.JasperInitializer.onStartup(JasperInitializer.java:101)
    at org.apache.catalina.core.StandardContext.startInternal(StandardContext.java:5178)
    at org.apache.catalina.util.LifecycleBase.start(LifecycleBase.java:150)
    at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1419)
    at org.apache.catalina.core.ContainerBase$StartChild.call(ContainerBase.java:1409)
    at java.util.concurrent.FutureTask.run(FutureTask.java:266)
    at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1142)
    at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)
    at java.lang.Thread.run(Thread.java:745)

任何帮助,将不胜感激。

4

3 回答 3

41

根本原因:

根据Tomcat Wiki,Servlet 3.0 规范要求在服务器启动期间扫描 Jar。

Tomcat 正在使用 org.apache.tomcat.util.scan。StandardJarScanner用于此目的。

来自StandardJarScanner的 javadoc 。

默认的 JarScanner 实现扫描 WEB-INF/lib目录,然后是提供的类加载器,然后处理类加载器层次结构。此实现足以满足 Servlet 3.0 规范的要求,并提供许多 Tomcat 特定的扩展。扩展是:

  • 扫描类加载器层次结构(默认启用) 测试所有文件以查看它们是否为 JAR(默认禁用)

  • 测试所有目录以查看它们是否是爆炸的 JAR(默认禁用)

  • 所有的扩展都可以通过配置来控制

解决方案1:特定于 Spring Boot。

我们可以禁用此 jar 扫描。

我通过在 application-xxx.properties 文件中添加以下属性来禁用它。此属性是Spring Boot 特定的。

# Comma-separated list of additional patterns that match jars to ignore for TLD scanning.    
server.tomcat.additional-tld-skip-patterns=*.jar

您可以在此处找到Tomcat 的类似属性

这些属性可用于配置传统的tomcat(spring boot)应用程序。

解决方案2:特定于弹簧

您可以为清单文件禁用 JarScanner,如下所示。

@Bean
public EmbeddedServletContainerFactory embeddedServletContainerFactory() {
  return new TomcatEmbeddedServletContainerFactory() {
    @Override
    protected void postProcessContext(Context context) {
      ((StandardJarScanner) context.getJarScanner()).setScanManifest(false);
    }
  };
}

解决方案3:传统的独立Tomcat:

<Context>
  ...
  <JarScanner scanManifest="false"/>
  ...
</Context>

参考:Jar Scanner 组件

于 2017-04-07T14:18:41.087 回答
17

只是为了改进 Sundaraj 的发现……完全禁用 TLD 扫描将破坏JSP/JSTL 支持。

问题是类路径本身没问题,只有 Tomcat额外扫描每个 Jar 的清单文件,并且由于使用 Maven,每个 Jar 都在自己的目录中,这会生成无意义的路径(可能从 Eclipse 运行?)。

因此,如果您想继续将 JSP 与 JSTL 一起使用,您应该只禁用清单扫描。

对于 Spring Boot 2.0,将其添加到应用程序的配置中:

  @Bean
  public TomcatServletWebServerFactory tomcatFactory() {
    return new TomcatServletWebServerFactory() {
      @Override
      protected void postProcessContext(Context context) {
        ((StandardJarScanner) context.getJarScanner()).setScanManifest(false);
      }
    };
  }
于 2018-09-07T20:31:21.120 回答
1

JarScannerFactory加载了我们需要在这里配置的 StandardJarScanner ,所以这个(对于 spring boot 2.1.8)也可以工作。

import org.apache.tomcat.JarScanner;
import org.apache.tomcat.util.scan.StandardJarScanner;
import org.springframework.boot.web.servlet.ServletContextInitializer;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class WebContextConfiguration {

    @Bean
    public ServletContextInitializer servletContextInitializer() {
        return context -> context.setAttribute(
                JarScanner.class.getName(),
                new StandardJarScanner() {{
                    setScanManifest(false);
                }}
        );
    }
}
于 2020-03-13T22:46:01.083 回答