Java Web Service REST 正确设计最佳实践

Posted

技术标签:

【中文标题】Java Web Service REST 正确设计最佳实践【英文标题】:Java Webservice REST proper design best pracice 【发布时间】:2011-09-08 13:07:25 【问题描述】:

我用 Java 开发了一个 web 服务,它作为 servlet 在 tomcat 上运行,应用程序的参数通过 get 请求(例如 servlet?method=search&query=searchterm123)提供给 servlet,servlet 识别方法和查询已定义,并且在出现错误的情况下返回一个字符串,该字符串手动包装在我通过this.writer.println(answer); 硬编码的 xml 代码中。如果方法正确,则实例化一个新类,该类进行搜索,然后返回一个对象,该对象 XStream 为我转换为 XML,然后我再次将其发送回客户端,并将 println 包装到我的 xml 开销中,该开销再次被硬编码 String @ 987654322@.

显然,这很有效,也很有效,但这远非优雅。我非常简要地研究了 Apache CXF,还研究了 RESTlet 或 JBOSS RestEasy 之类的东西,我发现后两者非常符合我的要求。我在 CXF 上找到的每个教程都包含 Spring 和 Maven,这让我有点不知所措。您对我应该如何将肮脏的 hack 变成漂亮的应用程序有什么建议吗?

【问题讨论】:

【参考方案1】:

我认为使用 JAX-RS 框架(如 CXF 和 Resteasy,还有Jersey)是您的解决方案。关于序列化为 XML,也许看看 JAXB(例如,也包含在 Jersey 中)。它应该有助于自动序列化任何实体结构。

关于此类应用程序的复杂性:它应始终取决于您使用的基础架构。如果它只是一个简单 Java EE 服务器,最好使用该供应商的实现(Glassfish 的 Jersey,JBoss 的 Resteasy)。否则,只需使用您熟悉和熟悉的构建系统。例如,您可以轻松地将 Maven 依赖项替换为 Ant 和 Ivy。

【讨论】:

【参考方案2】:

我可以推荐CXF;我发现学习它的教程非常容易,尤其是当我咬紧牙关并使用 Maven 来管理依赖项时(尽管它实际上与 CXF 和 Spring 所做的一切都正交)。

但是为了使用 CXF,我真的推荐使用 Spring。您不必使用 all 的 Spring; CXF 网站上的简单教程足以让您开始学习。如果您已经拥有实际实现已经完成的事情的代码,并且从代码中分离出来以解析传入的 URL、将响应呈现为 XML 等,则尤其如此;这就是 CXF(通常是 JAXB)将为您处理的部分。

为了提供帮助,这里有一个非常简单的示例(为简洁起见,省略了导入)。 我知道它看起来很复杂,但几乎所有的后半部分都包含你写一次然后不再真正接触的东西;当您构建以处理您的真实代码时,您可以做很多事情而不必完全关注框架代码。 一、接口定义(包括XML类型模型):

public interface Foo 
    @Path("/") @GET @Produces("application/xml");
    FooDesc getDescription(@Context UriInfo ui);
    @Path("id")
    FooItem getFoo(@PathParam("id") String id);


@Path("/")
public interface FooItem 
    @GET @Produces("application/xml")
    FooItemContents getContents();
    @PUT @Consumes("application/xml")
    void setContents(FooItemContents desc);
    @DELETE
    Response delete();


// These classes are purely structural holders; no behavior.

@XmlRootElement @XmlType
public class FooDesc 
    @XmlElement
    public List<URI> foo;


@XmlRootElement @XmlType
public class FooItemContents 
    @XmlElement
    String bar;

接下来是实现类:

public class FooImpl implements Foo 
    public FooDesc getDescription(UriInfo ui) 
        FooDesc desc = new FooDesc();
        desc.foo = new ArrayList<URI>();
        UriBuilder ub = ui.getAbsolutePathBuilder().path("id");
        for (String id : realGetIdList())  // Hook to your code here!
            desc.foo.add(ub.build(id));
        return desc;
    
    public FooItem getFoo(String id) 
        final RealFoo r = realGetById(id); // Hook to your code here!
        return new FooItem() 
            public FooItemContents getContents() 
                FooItemContents contents = new FooItemContents();
                contents.bar = r.getBar(); // Hook to your code here!
                return contents;
            
            public void setContents(FooItemContents desc) 
                r.setBar(desc.bar);        // Hook to your code here!
            
            public Response delete() 
                r.close();                 // Hook to your code here!
                return Response.noContent().build(); // Return a simple HTTP 204
            
        ;
    

现在,在 Spring 级别使用 WEB-INF/beans.xml 将其连接起来:

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:jaxrs="http://cxf.apache.org/jaxrs"
    xsi:schemaLocation="
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
        http://cxf.apache.org/jaxrs http://cxf.apache.org/schemas/jaxrs.xsd">

  <!-- Instantiate and connect the service framework -->
  <import resource="classpath:META-INF/cxf/cxf.xml" />
  <import resource="classpath:META-INF/cxf/cxf-extension-jaxrs-binding.xml" />
  <import resource="classpath:META-INF/cxf/cxf-servlet.xml" />
  <jaxrs:server id="customerService" address="/">
    <jaxrs:serviceBeans>
      <ref bean="fooBean" />
    </jaxrs:serviceBeans>
  </jaxrs:server>

  <!-- Instantiate your implementation class -->
  <bean id="fooBean" class="your.package.FooImpl" />
</beans>

现在,将其全部连接为 web 应用程序,web.xml

<web-app>
  <!-- Magic to make Spring work and build the rest for us -->
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>WEB-INF/beans.xml</param-value>
  </context-param>
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>

  <!-- Make CXF implement the servlet for us -->
  <servlet>
    <servlet-name>CXFServlet</servlet-name>
    <display-name>CXF Servlet</display-name>
    <servlet-class>org.apache.cxf.transport.servlet.CXFServlet</servlet-class>
    <load-on-startup>1</load-on-startup>
  </servlet>
  <!-- You *must* use this servlet mapping or Bad Things Happen. -->
  <servlet-mapping>
    <servlet-name>CXFServlet</servlet-name>
    <url-pattern>/*</url-pattern>
  </servlet-mapping>
</web-app>

现在构建一个 WAR,包括 CXF 站点上列出的所有库,将这些位放在正确的位置(您不需要 Maven,但最终它会更容易)并部署。这应该有效(即,你现在已经足够危险了!)

【讨论】:

我已将您的代码添加到我使用 cxf 原型创建的 Maven 项目中。不幸的是我得到了几十个错误,你能添加导入吗?另外我想知道例如 RealFoo 在 FooImpl.java 的第 15 行中,这个类从未定义过。

以上是关于Java Web Service REST 正确设计最佳实践的主要内容,如果未能解决你的问题,请参考以下文章

rest service技术选型

怎样更好的设计你的REST API之基于REST架构的Web Service设计及REST框架实现

Web Service和REST(下)

2.Spring构建REST Web Service

Rest,Rest Api,Web Service,RestFul Api之间的区别[关闭]

WCF Web API WCF REST 和 Web Service 的区别