我正在尝试使用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类的两个不同版本有关。当我只导入一个时,为什么会这样?
答案 0 :(得分:1)
昨天我使用Maven而不是Gradle遇到了这个问题。 你是对的,问题是由于在同一个包中有两个具有相同类名的jar:java.ws.rs.core.Application。
我能够通过搜索依赖关系树并找到哪个jar的依赖关系带来了应用程序的泽西版本并使用Maven的排除标记将其排除来修复它。
在你的情况下很明显:类加载器看到两个版本的java.ws.rs.core.Application并从org.glassfish.jersey.core中选择一个不是你想要的版本。
至于为什么它在Eclipse而不是在IDEA中工作,类加载器不保证加载类的顺序,使用两个不同的类加载器可能以不同的顺序加载类,如果有重复将导致不同的行为。