`
hawkmsg
  • 浏览: 6478 次
社区版块
存档分类
最新评论

转发-JAX-RS入门 一 :基础

阅读更多
来自:http://liugang594.iteye.com/blog/1491434


简介
JAX-RS是一套用java实现REST服务的规范,提供了一些标注将一个资源类,一个POJOJava类,封装为Web资源。标注包括:
@Path,标注资源类或方法的相对路径
@GET,@PUT,@POST,@DELETE,标注方法是用的HTTP请求的类型
@Produces,标注返回的MIME媒体类型
@Consumes,标注可接受请求的MIME媒体类型
@PathParam,@QueryParam,@HeaderParam,@CookieParam,@MatrixParam,@FormParam,分别标注方法的参数来自于HTTP请求的不同位置,例如@PathParam来自于URL的路径,@QueryParam来自于URL的查询参数,@HeaderParam来自于HTTP请求的头信息,@CookieParam来自于HTTP请求的Cookie。
目前JAX-RS的实现包括:
Apache CXF,开源的Web服务框架。
Jersey, 由Sun提供的JAX-RS的参考实现。
RESTEasy,JBoss的实现。
Restlet,由Jerome Louvel和Dave Pawson开发,是最早的REST框架,先于JAX-RS出现。
Apache Wink,一个Apache软件基金会孵化器中的项目,其服务模块实现JAX-RS规范
(以上来自:http://zh.wikipedia.org/wiki/JAX-RS)

装备
本文使用的工具有:
Eclipse-jee-helios
Java-1.6.0_26
apache-tomcat-6.0.30
SoapUI-3.6
使用到的外部jar包有(必须的部分,需要加到Web容器中)
neethi-3.0.2.jar
jsr311-api-1.1.1.jar
cxf-bundle-2.6.0.jar
使用到的外部jar包有(可选的部分,当且仅当作为一个独立的application运行时)
jetty-http-7.5.4.v20111024.jar
jetty-io-7.5.4.v20111024.jar
jetty-server-7.5.4.v20111024.jar
jetty-util-7.5.4.v20111024.jar
jetty-continuation-7.5.4.v20111024.jar
wsdl4j-1.6.2.jar
准备
(以下例子来自: Oreilly - RESTful Java with JAX-RS (12-2009) (ATTiCA).pdf)

创建工程
为了后续顺利进行,首先在eclipse上先创建一个Dynamic Web Project,完成以后,一个符合war结构的工程目录会自动生成,之后可以很简单的导出为war文件,其中需要把以下jar包放到 /WebContent/WEB-INF/lib 里:
neethi-3.0.2.jar
jsr311-api-1.1.1.jar
cxf-bundle-2.6.0.jar
另外,在工程目录下,新建一个 lib 文件夹用来存放以下可选的jar包:

jetty-http-7.5.4.v20111024.jar
jetty-io-7.5.4.v20111024.jar
jetty-server-7.5.4.v20111024.jar
jetty-util-7.5.4.v20111024.jar
jetty-continuation-7.5.4.v20111024.jar
wsdl4j-1.6.2.jar
最后一步就是把所有这9个jar都加到工程的build path里去,这样工程就准备好了。

定义服务
这里要实现一个简单的REST服务用于对客户进行管理,包括:
创建客户
查看客户
更新客户
首先给出对应的于这些操作的服务接口:
Java代码 
import java.io.InputStream; 
 
import javax.ws.rs.Consumes; 
import javax.ws.rs.GET; 
import javax.ws.rs.POST; 
import javax.ws.rs.PUT; 
import javax.ws.rs.Path; 
import javax.ws.rs.PathParam; 
import javax.ws.rs.Produces; 
import javax.ws.rs.core.Response; 
import javax.ws.rs.core.StreamingOutput; 
 
@Path("/customers") 
public interface CustomerResource { 
 
    @POST 
    @Consumes("application/xml") 
    public Response createCustomer(InputStream is); 
 
    @GET 
    @Path("{id}") 
    @Produces("application/xml") 
    public StreamingOutput getCustomer(@PathParam("id") int id); 
 
    @PUT 
    @Path("{id}") 
    @Consumes("application/xml") 
    public void updateCustomer(@PathParam("id") int id, InputStream is) ; 


令人惊奇的是,这个接口已经包含了所有实现我们既定目标的关键部分:
@Path: 定义服务路径,接口中定义的整个服务的顶级路径为"/customers ",方法对应的服务路径为接口路径加方法定义的Path值,如果未定义,则用接口路径,例如getCustomer()的服务路径为:" /customers/{id} "。所以此REST对外服务路径都是 服务的上下文路径/customers/ 子级目录,
@POST,@GET,@PUT:标注方法所支持HTTP请求的类型 (参考上面的说明)
@Produces,@Consumes:标注方法支持或返回的请求MIME类型。
由上可以看到,每个方法被调用的条件如下:
createConsumer(): 请求HTTP方法为POST;请求MIME类型为application/xml;请求路径为: 上下文路径/customers
getCustomer(): 请求的HTTP方法为GET;请求的MIME类型为application/xml;请求的路径为: 上下文路径/customers/{id}
注: {id}为某个存在(或不存在)customer的编号
updateCustomer(): 请求的HTTP方法为PUT;请求的MIME类型为application/xml;请求的路径: 上下文路径/customers/{id}
注: {id}为某个存在(或不存在)customer的编号
一个好的实现方法是将REST服务的定义和实现分开,这样代码的结构简洁、清晰,在后期也可以很方便的进行实现的替换和服务定义的修改。

下面就是添加实现部分:
Java代码 
public class CustomerResourceService implements CustomerResource{ 
    private Map<Integer, Customer> customerDB = new ConcurrentHashMap<Integer, Customer>(); 
    private AtomicInteger idCounter = new AtomicInteger(); 
 
    public Response createCustomer(InputStream is) { 
        Customer customer = readCustomer(is); 
        customer.setId(idCounter.incrementAndGet()); 
        customerDB.put(customer.getId(), customer); 
        System.out.println("Created customer " + customer.getId()); 
        return Response.created(URI.create("/customers/" + customer.getId())) 
                .build(); 
    } 
 
    public StreamingOutput getCustomer(int id) { 
        final Customer customer = customerDB.get(id); 
        if (customer == null) { 
            throw new WebApplicationException(Response.Status.NOT_FOUND); 
        } 
        return new StreamingOutput() { 
            public void write(OutputStream outputStream) throws IOException, 
                    WebApplicationException { 
                outputCustomer(outputStream, customer); 
            } 
        }; 
    } 
 
    public void updateCustomer(int id, InputStream is) { 
        Customer update = readCustomer(is); 
        Customer current = customerDB.get(id); 
        if (current == null) 
            throw new WebApplicationException(Response.Status.NOT_FOUND); 
        current.setFirstName(update.getFirstName()); 
        current.setLastName(update.getLastName()); 
        current.setStreet(update.getStreet()); 
        current.setState(update.getState()); 
        current.setZip(update.getZip()); 
        current.setCountry(update.getCountry()); 
    } 
 
    protected void outputCustomer(OutputStream os, Customer cust) 
            throws IOException { 
        PrintStream writer = new PrintStream(os); 
        writer.println("<customer id=\"" + cust.getId() + "\">"); 
        writer.println(" <first-name>" + cust.getFirstName() + "</first-name>"); 
        writer.println(" <last-name>" + cust.getLastName() + "</last-name>"); 
        writer.println(" <street>" + cust.getStreet() + "</street>"); 
        writer.println(" <city>" + cust.getCity() + "</city>"); 
        writer.println(" <state>" + cust.getState() + "</state>"); 
        writer.println(" <zip>" + cust.getZip() + "</zip>"); 
        writer.println(" <country>" + cust.getCountry() + "</country>"); 
        writer.println("</customer>"); 
    } 
 
    protected Customer readCustomer(InputStream is) { 
        try { 
            DocumentBuilder builder = DocumentBuilderFactory.newInstance() 
                    .newDocumentBuilder(); 
            Document doc = builder.parse(is); 
            Element root = doc.getDocumentElement(); 
            Customer cust = new Customer(); 
            if (root.getAttribute("id") != null 
                    && !root.getAttribute("id").trim().equals("")) { 
                cust.setId(Integer.valueOf(root.getAttribute("id"))); 
            } 
            NodeList nodes = root.getChildNodes(); 
            for (int i = 0; i < nodes.getLength(); i++) { 
                Node item = nodes.item(i); 
                if(!(item instanceof Element)){ 
                    continue; 
                } 
                Element element = (Element) nodes.item(i); 
                if (element.getTagName().equals("first-name")) { 
                    cust.setFirstName(element.getTextContent()); 
                } else if (element.getTagName().equals("last-name")) { 
                    cust.setLastName(element.getTextContent()); 
                } else if (element.getTagName().equals("street")) { 
                    cust.setStreet(element.getTextContent()); 
                } else if (element.getTagName().equals("city")) { 
                    cust.setCity(element.getTextContent()); 
                } else if (element.getTagName().equals("state")) { 
                    cust.setState(element.getTextContent()); 
                } else if (element.getTagName().equals("zip")) { 
                    cust.setZip(element.getTextContent()); 
                } else if (element.getTagName().equals("country")) { 
                    cust.setCountry(element.getTextContent()); 
                } 
            } 
            return cust; 
        } catch (Exception e) { 
            throw new WebApplicationException(e, Response.Status.BAD_REQUEST); 
        } 
    } 


这些方法的实现都很直接,不细说,不过有一点需要特别注意的是:

最好不要在实现中混杂有服务的定义部分,例如@Path标签,@PathParam标签等等,如果想修改定义,最好是在接口中修改;或者如果想覆盖某个接口方法的某个annotation,则所有该接口方法的annotation定义都需要重写,而不能仅修改变化的。
分享到:
评论

相关推荐

    cxf-rt-frontend-jaxrs-3.0.1-API文档-中文版.zip

    赠送jar包:cxf-rt-frontend-jaxrs-3.0.1.jar; 赠送原API文档:cxf-rt-frontend-jaxrs-3.0.1-javadoc.jar; 赠送源代码:cxf-rt-frontend-jaxrs-3.0.1-sources.jar; 赠送Maven依赖信息文件:cxf-rt-frontend-jaxrs...

    jackson-jaxrs-json-provider-2.7.8-API文档-中文版.zip

    赠送jar包:jackson-jaxrs-json-provider-2.7.8.jar; 赠送原API文档:jackson-jaxrs-json-provider-2.7.8-javadoc.jar; 赠送源代码:jackson-jaxrs-json-provider-2.7.8-sources.jar; 赠送Maven依赖信息文件:...

    jackson-jaxrs-json-provider-2.7.8-API文档-中英对照版.zip

    赠送jar包:jackson-jaxrs-json-provider-2.7.8.jar; 赠送原API文档:jackson-jaxrs-json-provider-2.7.8-javadoc.jar; 赠送源代码:jackson-jaxrs-json-provider-2.7.8-sources.jar; 赠送Maven依赖信息文件:...

    fastjson-jaxrs-json-provider-0.3.1-API文档-中文版.zip

    赠送jar包:fastjson-jaxrs-json-provider-0.3.1.jar; 赠送原API文档:fastjson-jaxrs-json-provider-0.3.1-javadoc.jar; 赠送源代码:fastjson-jaxrs-json-provider-0.3.1-sources.jar; 赠送Maven依赖信息文件:...

    jackson-jaxrs-base-2.7.8-API文档-中英对照版.zip

    赠送jar包:jackson-jaxrs-base-2.7.8.jar; 赠送原API文档:jackson-jaxrs-base-2.7.8-javadoc.jar; 赠送源代码:jackson-jaxrs-base-2.7.8-sources.jar; 赠送Maven依赖信息文件:jackson-jaxrs-base-2.7.8.pom;...

    cxf-rt-frontend-jaxrs-2.7.16.zip

    2. JAX-RS 是 Java 平台上的 RESTful API,CXF 的 frontend-jaxrs 模块支持这一标准。 3. "cxf-rt-frontend-jaxrs-2.7.16.zip" 包含了 CXF 的 JAX-RS 实现,适合构建 RESTful 应用程序。 4. ph-commons 是一个跨版本...

    spring-boot-cxf-jaxrs:使用Spring Boot和CXF JAXRS快速入门

    快速入门使用Spring Boot来配置一个小的应用程序,其中包括启用了Swagger的CXF JAXRS端点。 重要的 该快速入门可以在2种模式下运行:在您的计算机和Kubernetes / OpenShift群集上独立运行 部署选项 您可以在以下...

    jackson-jaxrs-base-2.7.8-API文档-中文版.zip

    赠送jar包:jackson-jaxrs-base-2.7.8.jar; 赠送原API文档:jackson-jaxrs-base-2.7.8-javadoc.jar; 赠送源代码:jackson-jaxrs-base-2.7.8-sources.jar; 赠送Maven依赖信息文件:jackson-jaxrs-base-2.7.8.pom;...

    spring-boot-cxf-jaxrs:Spring Boot CXF 集成

    CXF 的 Spring Boot 集成,特别是 JAX-RS。 用法 将以下内容添加到您的pom.xml (在适当的情况下更新&lt;version&gt; ): &lt;groupId&gt;com.internetitem.spring&lt;/groupId&gt; &lt;artifactId&gt;spring-boot-cxf-jaxrs ...

    jackson-jaxrs-1.9.11.jar

    jackson-jaxrs-1.9.11.jar

    SpringMVC精品资源--JAX-RS &amp; SpringMVC supported maven buil.zip

    这个压缩包"SpringMVC精品资源--JAX-RS & SpringMVC supported maven build.zip"显然是一个关于这两个框架的学习资源集合,特别是针对使用Maven构建的项目。 首先,SpringMVC是Spring框架的一部分,它是基于模型-...

    jackson-jaxrs-providers:包含用于JSON,XML,YAML,Smile,CBOR格式的基于Jackson的JAX-RS提供程序的多模块项目

    概述这是一个多模块项目,包含用于以下数据格式的基于Jackson的JAX-RS提供程序:(二进制JSON) (另一种二进制JSON) 提供程序实现特定数据格式的JAX-RS MessageBodyReader和MessageBodyWriter处理程序。...

    resteasy-jaxrs-2.2 学习整理的例子

    标题 "resteasy-jaxrs-2.2 学习整理的例子" 暗示了我们即将探讨的是关于RESTful服务开发的一个案例,其中涉及到的主要技术是RESTEasy和JAX-RS 2.2。RESTEasy是Java EE平台上的一个开源框架,它实现了Java API for ...

    org.apache.servicemix.bundles.swagger-jaxrs-1.3.2_2.zip

    Swagger-JAXRS是一个用于构建RESTful API的工具,它允许开发者通过JAX-RS注解来描述API接口,进而生成交互式的API文档。这个版本可能已经被Apache ServiceMix整合和优化,以便在服务混合环境中更好地工作。 描述中...

    jboss-jaxrs-api_1.1_spec-1.0.0.Final.zip

    【标题】"jbosss-jaxrs-api_1.1_spec-1.0.0.Final.zip" 提供的是Java企业版(Java EE)中的一部分关键规范,即Java API for RESTful Web Services (JAX-RS) 1.1的实现。JAX-RS是Java平台上的一个标准接口,它允许...

    jax-rs jax-ws所需包,亲测可用

    **标题与描述解析:** 标题"jax-rs jax-ws所需包,亲测可用"表明这个压缩包包含了用于开发...通过以上步骤,开发者可以快速搭建一个能正确处理JAX-RS和JAX-WS服务的环境,无需手动解决依赖问题,大大提高了开发效率。

    jboss-jaxrs-2.0-api-1.0.0-6.el8.noarch(1).rpm

    官方离线安装包,亲测可用

    jackson-jaxrs-json-provider-2.2.4.zip

    标题中的"jackson-jaxrs-json-provider-2.2.4.zip"指示了这是一个与Jackson框架相关的库,特别地,它是Jackson JAX-RS JSON提供者的一个版本。Jackson是Java中广泛使用的JSON处理库,而JAX-RS是Java API for RESTful...

    cxf-rt-frontend-jaxrs-3.0.3.jar

    java运行依赖jar包

    cxf-bundle-jaxrs-2.4.7.zip

    【标题】"CXF-BUNDLE-JAXRS-2.4.7.zip" 是一个与Apache CXF相关的开源项目,它包含了一个JAX-RS(Java API for RESTful Web Services)的OSGi捆绑包。JAX-RS是Java平台上的标准,用于创建RESTful服务,而CXF是一个...

Global site tag (gtag.js) - Google Analytics