java.lang.ClassNotFoundException:javax.ws.rs.MessageProcessingException

Dan*_*lan 5 java tomcat cxf jax-rs jersey-client

我正在向Tomcat 7.0.57部署战争.此代码使用Jersey 2.x客户端与Rest端点进行通信,并使用CXF公开其自己的Rest端点(AKA为war的端点).

当我点击战争的一个端点时,代码似乎工作并从服务器的角度返回一个没有问题的响应,但客户端从tomcat返回500响应.这是错误:

java.lang.ClassNotFoundException: javax.ws.rs.MessageProcessingException
    org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1720)
    org.apache.catalina.loader.WebappClassLoader.loadClass(WebappClassLoader.java:1571)
    org.apache.cxf.jaxrs.impl.ResponseBuilderImpl.build(ResponseBuilderImpl.java:69)
    org.apache.cxf.jaxrs.interceptor.JAXRSOutInterceptor.processResponse(JAXRSOutInterceptor.java:137)
    org.apache.cxf.jaxrs.interceptor.JAXRSOutInterceptor.handleMessage(JAXRSOutInterceptor.java:86)
    org.apache.cxf.phase.PhaseInterceptorChain.doIntercept(PhaseInterceptorChain.java:272)
    org.apache.cxf.interceptor.OutgoingChainInterceptor.handleMessage(OutgoingChainInterceptor.java:77)
    org.apache.cxf.phase.PhaseInterceptorChain.doIntercept(PhaseInterceptorChain.java:272)
    org.apache.cxf.transport.ChainInitiationObserver.onMessage(ChainInitiationObserver.java:121)
    org.apache.cxf.transport.http.AbstractHTTPDestination.invoke(AbstractHTTPDestination.java:239)
    org.apache.cxf.transport.servlet.ServletController.invokeDestination(ServletController.java:248)
    org.apache.cxf.transport.servlet.ServletController.invoke(ServletController.java:222)
    org.apache.cxf.transport.servlet.ServletController.invoke(ServletController.java:153)
    org.apache.cxf.transport.servlet.CXFNonSpringServlet.invoke(CXFNonSpringServlet.java:167)
    org.apache.cxf.transport.servlet.AbstractHTTPServlet.handleRequest(AbstractHTTPServlet.java:286)
    org.apache.cxf.transport.servlet.AbstractHTTPServlet.doPost(AbstractHTTPServlet.java:206)
    javax.servlet.http.HttpServlet.service(HttpServlet.java:646)
    org.apache.cxf.transport.servlet.AbstractHTTPServlet.service(AbstractHTTPServlet.java:262)
    org.apache.tomcat.websocket.server.WsFilter.doFilter(WsFilter.java:52)
Run Code Online (Sandbox Code Playgroud)

奇怪的是,这个错误没有出现在tomcat日志中.

我做了一点研究,似乎这个类是jax-rs的一部分.在我的maven pom中,我已经包含了这种依赖:

    <dependency>
        <groupId>javax.ws.rs</groupId>
        <artifactId>javax.ws.rs-api</artifactId>
        <version>2.0.1</version>
    </dependency>
Run Code Online (Sandbox Code Playgroud)

但是这种依赖似乎没有这个类.这个类在这种依赖,虽然:

    <dependency>
        <groupId>javax.ws.rs</groupId>
        <artifactId>javax.ws.rs-api</artifactId>
        <version>2.0-m01</version>
    </dependency>
Run Code Online (Sandbox Code Playgroud)

我感到不舒服的降级到较低版本,这是一个里程碑.更重要的是,那个jar不包括我在我的代码中使用的类javax.ws.rs.client.ClientBuilder.

任何人都可以解释为什么我得到这个例外以及如何解决这个问题?


我的解决方法

我得出的结论是,这与泽西客户端和CXF相互干扰更多.我决定删除Jersey客户端并将其替换为CXF客户端.这些说明比官方说明要好得多:http://fandry.blogspot.com/2012/06/how-to-create-simple-cxf-based-jax-rs.html

import com.fasterxml.jackson.jaxrs.json.JacksonJaxbJsonProvider;
import org.apache.cxf.jaxrs.client.WebClient;
import java.util.ArrayList;
import java.util.List;

public class App 
{
    public static void main( String[] args )  throws Exception { 

     List<Object> providers = new ArrayList<Object>();
     providers.add( new JacksonJaxbJsonProvider() );

     WebClient client = WebClient.create("http://localhost:8080/poc_restapi_cxf/api", providers);
     client = client.accept("application/json").type("application/json").path("/order").query("id", "1");

     Order order = client.get(Order.class);
     System.out.println("Order:" + order.getCustomerName());

    }
}
Run Code Online (Sandbox Code Playgroud)

非常相似的API.我只是搜索并替换了一些方法名称,一切正常.

Sam*_*ays 2

我也遇到过这个问题。与您的项目要求不同,我必须让 Jersey 和 CXF 发挥良好作用。

最终的问题是在 FactoryFinder 类中,Jersey 使用该类来获取适当的运行时委托(用于各种辅助对象) - 这又具有一个导入有问题的类的类。

为了解决这个问题,您需要在资源文件夹中拥有一个文件(假设该文件导入到您的类路径文件夹中),其路径 META-INF/services/javax.ws.rs.ext.RuntimeDelegate 包含以下值:

org.glassfish.jersey.internal.RuntimeDelegateImpl
Run Code Online (Sandbox Code Playgroud)

这应该可以解决你的问题。