IntelliJ中的NoSuchMethodError,但不是Eclipse

时间:2015-04-26 16:55:40

标签: java eclipse intellij-idea gradle

我正在尝试使用Jetty Server运行Jersey REST服务。当我在IntelliJ中运行时,我收到以下错误:

java.lang.NoSuchMethodError: javax.ws.rs.core.Application.getProperties()Ljava/util/Map;
    at org.glassfish.jersey.server.ApplicationHandler.<init>(ApplicationHandler.java:309)
    at org.glassfish.jersey.servlet.WebComponent.<init>(WebComponent.java:338)
    at org.glassfish.jersey.servlet.ServletContainer.init(ServletContainer.java:171)
    at org.glassfish.jersey.servlet.ServletContainer.init(ServletContainer.java:363)
    at javax.servlet.GenericServlet.init(GenericServlet.java:244)
    etc...
    at service.WebserviceRunner.start(WebserviceRunner.java:37)

如果我在Eclipse中运行相同的项目它运行正常。为什么IDE会有所作为呢?

WebserviceRunner.java

import org.eclipse.jetty.server.Server;
import org.eclipse.jetty.server.handler.HandlerCollection;
import org.eclipse.jetty.servlet.ServletHolder;
import org.eclipse.jetty.webapp.WebAppContext;
import org.glassfish.jersey.server.ServerProperties;
import org.glassfish.jersey.servlet.ServletContainer;

public class WebserviceRunner {

    private Server server;
    private int port;
    private String host;

    public WebserviceRunner(String host, int port) {
        this.server = new Server(port);
        this.host = host;
        this.port = port;
    }

    public void start() throws Exception {
        HandlerCollection handlers = new HandlerCollection();
        WebAppContext handler = new WebAppContext();

        handler.setContextPath("/");
        handler.setResourceBase("./");
        handler.setClassLoader(Thread.currentThread().getContextClassLoader());

        ServletHolder restServlet = handler.addServlet(ServletContainer.class,  "/*");
        restServlet.setInitOrder(0);
        restServlet.setInitParameter(ServerProperties.PROVIDER_PACKAGES,"resource");

        handlers.addHandler(handler);

        server.setHandler(handlers);
        server.start();
        System.out.println("API started... at 'http://" + getHostAndPort() + "'");
    }

    private String getHostAndPort() {
        return host + ":" + port;
    }

    public static void main(String[] args) throws Exception {
        new WebserviceRunner("localhost", 8315).start();
    }
}

的build.gradle

apply plugin: 'java'
apply plugin: 'jetty'
apply plugin: 'eclipse'
apply plugin: 'idea'
apply plugin: 'war'

repositories {
    mavenCentral()
}

dependencies {
    compile 'org.glassfish.jersey.core:jersey-server:2.17'
    compile 'org.glassfish.jersey.containers:jersey-container-servlet-core:2.17'

    compile 'org.javassist:javassist:3.15.0-GA'
    compile 'javax.ws.rs:javax.ws.rs-api:2.0.1'
    compile 'javax.inject:javax.inject:1'
    compile 'com.google.code.gson:gson:2.2.2'
    compile 'com.google.guava:guava:10.0'
    compile 'com.fasterxml.jackson.jaxrs:jackson-jaxrs-json-provider:2.3.0'
    testCompile group: 'junit', name: 'junit', version: '4.11'

    def jettyVersion = '7.2.2.v20101205';
    compile "org.eclipse.jetty:jetty-server:${jettyVersion}"
    compile "org.eclipse.jetty:jetty-webapp:${jettyVersion}"
    compile "org.eclipse.jetty:jetty-servlet:${jettyVersion}"
    compile "org.eclipse.jetty:jetty-servlets:${jettyVersion}"
    compile "org.eclipse.jetty:jetty-annotations:${jettyVersion}"
    compile 'org.mortbay.jetty:jsp-2.1-glassfish:2.1.v20100127'
    compile 'javax.servlet:jstl:1.2'
    compile 'javax.servlet:javax.servlet-api:3.0.1'
}

注意:我认为这与我的类路径上具有相同Application类的两个不同版本有关。当我只导入一个时,为什么会这样?

enter image description here

1 个答案:

答案 0 :(得分:1)

昨天我使用Maven而不是Gradle遇到了这个问题。 你是对的,问题是由于在同一个包中有两个具有相同类名的jar:java.ws.rs.core.Application。

我能够通过搜索依赖关系树并找到哪个jar的依赖关系带来了应用程序的泽西版本并使用Maven的排除标记将其排除来修复它。

在你的情况下很明显:类加载器看到两个版本的java.ws.rs.core.Application并从org.glassfish.jersey.core中选择一个不是你想要的版本。

至于为什么它在Eclipse而不是在IDEA中工作,类加载器不保证加载类的顺序,使用两个不同的类加载器可能以不同的顺序加载类,如果有重复将导致不同的行为。