我有一个 Java 服务器应用程序,它使用 CXF 来提供 SOAP 和 REST Web 服务。目前,它使用 JAX-B 的引用实现进行 XML 编码/解码,但我已将其配置为用 Jackson 替换 Jettison 进行 JSON 编码/解码。我使用 Spring 进行 DI 和应用程序上下文配置。

REST Web 服务配置片段如下所示:

web.xml

<servlet>
    <display-name>Myapp REST Services</display-name>
    <servlet-name>MyappWebServices</servlet-name>
    <servlet-class>org.apache.cxf.transport.servlet.CXFServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
</servlet>

<servlet-mapping>
    <servlet-name>MyappWebServices</servlet-name>
    <url-pattern>/services/*</url-pattern>
</servlet-mapping>

applicationContext.xml
<import resource="classpath:META-INF/cxf/cxf.xml" />
<import resource="classpath:META-INF/cxf/cxf-servlet.xml" />

<bean id="jsonProvider" class="org.codehaus.jackson.jaxrs.JacksonJsonProvider" />

<jaxrs:server id="myappCoreSvcRest" address="/rest">
    <jaxrs:serviceBeans>
        <ref bean="fooService" />
        <ref bean="barService" />
    </jaxrs:serviceBeans>

    <jaxrs:providers>
        <ref bean="jsonProvider" />
    </jaxrs:providers>
</jaxrs:server>

此配置有效,并将根据 HTTP 接受 header 返回 XML 或 JSON。我喜欢这个配置的地方在于它基于 Spring,创建和使用备用 JSON 编码器非常容易。可以找到有关配置 CXF 的详细信息 here

我的问题是现在我有一个新的(附加的)REST web 服务要提供,我想为这个新的 web 服务使用不同的 JAX-B XML 绑定(bind)。我知道 MOXy 可以做到这一点,但我无法弄清楚如何配置 CXF 端点,以便它使用 MOXy 进行编码/解码(以及如何将我的自定义 XML 映射文件告诉 Moxy)。我还希望这个新的 Web 服务根据 Accept header 返回 XML 或 JSON。我还读到 MOXy 2.4+ 也可以处理!

理想情况下,我可以将 MOXy 用于这个新端点,而不会影响其他现有的 servlet。

最佳答案

注意: 我是 EclipseLink JAXB (MOXy) 的负责人,也是 JAXB (JSR-222) 专家组的成员。

我暂时不知道 CXF 的确切配置,但在下面我提供了一些将 MOXy 与 Spring 结合使用的链接。请随时 contact me ,我可以帮助您实现这一点:

  • http://blog.bdoughan.com/p/contact_01.html



  • 在 JAX-RS 实现中使用 MOXy 时,您可以使用 ContextResolver 从 MOXy 的外部映射文件进行引导:
    package blog.bindingfile.jaxrs;
    
    import java.io.*;
    import java.util.*;
    import javax.ws.rs.Produces;
    import javax.ws.rs.ext.*;
    import javax.xml.bind.*;
    import org.eclipse.persistence.jaxb.JAXBContextFactory;
    
    import blog.bindingfile.Customer;
    
    @Provider
    @Produces({"application/xml", "application/json"})
    public class CustomerContextResolver implements ContextResolver<JAXBContext> {
    
        private JAXBContext jc;
    
        public CustomerContextResolver() {
            ClassLoader cl = Customer.class.getClassLoader();
            InputStream bindings =
                cl.getResourceAsStream("blog/bindingfile/binding.xml");
            try {
                Map<String, Object> props = new HashMap<String, Object>(1);
                props.put(JAXBContextFactory.ECLIPSELINK_OXM_XML_KEY, bindings);
                jc = JAXBContext.newInstance(new Class[] {Customer.class} , props);
            } catch(JAXBException e) {
                throw new RuntimeException(e);
            } finally {
                try {
                    bindings.close();
                } catch(IOException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    
        public JAXBContext getContext(Class<?> clazz) {
            if(Customer.class == clazz) {
                return jc;
            }
            return null;
        }
    
    }
    

    对于复杂示例
  • http://blog.bdoughan.com/2011/04/moxys-xml-metadata-in-jax-rs-service.html

  • 有关在 Spring 中使用 MOXy 的更多信息
  • http://wiki.eclipse.org/EclipseLink/Examples/MOXy/Spring




  • 是的,JSON 绑定(bind)正在添加到 EclipseLink 2.4。要在您的应用程序中利用它,创建 MessageBodyReaderMessageBodyWriter 应该很简单:
    package org.example;
    
    import java.io.*;
    import java.lang.annotation.Annotation;
    import java.lang.reflect.*;
    import javax.xml.transform.stream.StreamSource;
    
    import javax.ws.rs.*;
    import javax.ws.rs.core.*;
    import javax.ws.rs.ext.*;
    import javax.xml.bind.*;
    
    @Provider
    @Produces(MediaType.APPLICATION_JSON)
    @Consumes(MediaType.APPLICATION_JSON)
    public class MOXyJSONProvider implements
        MessageBodyReader<Object>, MessageBodyWriter<Object>{
    
        @Context
        protected Providers providers;
    
        public boolean isReadable(Class<?> type, Type genericType,
            Annotation[] annotations, MediaType mediaType) {
            return true;
        }
    
        public Object readFrom(Class<Object> type, Type genericType,
                Annotation[] annotations, MediaType mediaType,
                MultivaluedMap<String, String> httpHeaders, InputStream entityStream)
                throws IOException, WebApplicationException {
                try {
                    Class<?> domainClass = getDomainClass(genericType);
                    Unmarshaller u = getJAXBContext(domainClass, mediaType).createUnmarshaller();
                    u.setProperty("eclipselink.media-type", mediaType.toString());
                    u.setProperty("eclipselink.json.include-root", false);
                    return u.unmarshal(new StreamSource(entityStream), domainClass).getValue();
                } catch(JAXBException jaxbException) {
                    throw new WebApplicationException(jaxbException);
                }
        }
    
        public boolean isWriteable(Class<?> type, Type genericType,
            Annotation[] annotations, MediaType mediaType) {
            return true;
        }
    
        public void writeTo(Object object, Class<?> type, Type genericType,
            Annotation[] annotations, MediaType mediaType,
            MultivaluedMap<String, Object> httpHeaders,
            OutputStream entityStream) throws IOException,
            WebApplicationException {
            try {
                Class<?> domainClass = getDomainClass(genericType);
                Marshaller m = getJAXBContext(domainClass, mediaType).createMarshaller();
                m.setProperty("eclipselink.media-type", mediaType.toString());
                m.setProperty("eclipselink.json.include-root", false);
                m.marshal(object, entityStream);
            } catch(JAXBException jaxbException) {
                throw new WebApplicationException(jaxbException);
            }
        }
    
        public long getSize(Object t, Class<?> type, Type genericType,
            Annotation[] annotations, MediaType mediaType) {
            return -1;
        }
    
        private JAXBContext getJAXBContext(Class<?> type, MediaType mediaType)
            throws JAXBException {
            ContextResolver<JAXBContext> resolver
                = providers.getContextResolver(JAXBContext.class, mediaType);
            JAXBContext jaxbContext;
            if(null == resolver || null == (jaxbContext = resolver.getContext(type))) {
                return JAXBContext.newInstance(type);
            } else {
                return jaxbContext;
            }
        }
    
        private Class<?> getDomainClass(Type genericType) {
            if(genericType instanceof Class) {
                return (Class<?>) genericType;
            } else if(genericType instanceof ParameterizedType) {
                return (Class<?>) ((ParameterizedType) genericType).getActualTypeArguments()[0];
            } else {
                return null;
            }
        }
    
    }
    

    您也可以创建 JSONProvider 的扩展:
  • http://cxf.apache.org/apidocs/org/apache/cxf/jaxrs/provider/JSONProvider.html

  • 了解更多信息
  • MOXy as Your JAX-RS JSON Provider - Client Side
  • MOXy as Your JAX-RS JSON Provider - Server Side
  • Specifying EclipseLink MOXy as Your JAXB Provider
  • 关于java - 使用 Spring 配置 CXF 以使用 MOXY 进行 XML 编码/解码,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/8980204/

    10-12 02:06