zoukankan      html  css  js  c++  java
  • JAX-RS入门 一 :基础

    官方文档:https://jersey.java.net/documentation/latest/jaxrs-resources.html

    简介

    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的实现包括:

    (以上来自: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代码  收藏代码
    1. import java.io.InputStream;  
    2.   
    3. import javax.ws.rs.Consumes;  
    4. import javax.ws.rs.GET;  
    5. import javax.ws.rs.POST;  
    6. import javax.ws.rs.PUT;  
    7. import javax.ws.rs.Path;  
    8. import javax.ws.rs.PathParam;  
    9. import javax.ws.rs.Produces;  
    10. import javax.ws.rs.core.Response;  
    11. import javax.ws.rs.core.StreamingOutput;  
    12.   
    13. @Path("/customers")  
    14. public interface CustomerResource {  
    15.   
    16.     @POST  
    17.     @Consumes("application/xml")  
    18.     public Response createCustomer(InputStream is);  
    19.   
    20.     @GET  
    21.     @Path("{id}")  
    22.     @Produces("application/xml")  
    23.     public StreamingOutput getCustomer(@PathParam("id") int id);  
    24.   
    25.     @PUT  
    26.     @Path("{id}")  
    27.     @Consumes("application/xml")  
    28.     public void updateCustomer(@PathParam("id") int id, InputStream is) ;  
    29. }  

     令人惊奇的是,这个接口已经包含了所有实现我们既定目标的关键部分:

    1. @Path: 定义服务路径,接口中定义的整个服务的顶级路径为"/customers ",方法对应的服务路径为接口路径加方法定义的Path值,如果未定义,则用接口路径,例如getCustomer()的服务路径为:" /customers/{id}"。所以此REST对外服务路径都是 服务的上下文路径/customers/ 子级目录,
    2. @POST,@GET,@PUT:标注方法所支持HTTP请求的类型 (参考上面的说明)
    3. @Produces,@Consumes:标注方法支持或返回的请求MIME类型。

    由上可以看到,每个方法被调用的条件如下:

    1. createConsumer(): 请求HTTP方法为POST;请求MIME类型为application/xml;请求路径为: 上下文路径/customers 
    2. getCustomer(): 请求的HTTP方法为GET;请求的MIME类型为application/xml;请求的路径为: 上下文路径/customers/{id} 
      注: {id}为某个存在(或不存在)customer的编号
    3. updateCustomer(): 请求的HTTP方法为PUT;请求的MIME类型为application/xml;请求的路径: 上下文路径/customers/{id}
      注: {id}为某个存在(或不存在)customer的编号

    一个好的实现方法是将REST服务的定义和实现分开,这样代码的结构简洁、清晰,在后期也可以很方便的进行实现的替换和服务定义的修改。

    下面就是添加实现部分:

    Java代码  收藏代码
    1. public class CustomerResourceService implements CustomerResource{  
    2.     private Map<Integer, Customer> customerDB = new ConcurrentHashMap<Integer, Customer>();  
    3.     private AtomicInteger idCounter = new AtomicInteger();  
    4.   
    5.     public Response createCustomer(InputStream is) {  
    6.         Customer customer = readCustomer(is);  
    7.         customer.setId(idCounter.incrementAndGet());  
    8.         customerDB.put(customer.getId(), customer);  
    9.         System.out.println("Created customer " + customer.getId());  
    10.         return Response.created(URI.create("/customers/" + customer.getId()))  
    11.                 .build();  
    12.     }  
    13.   
    14.     public StreamingOutput getCustomer(int id) {  
    15.         final Customer customer = customerDB.get(id);  
    16.         if (customer == null) {  
    17.             throw new WebApplicationException(Response.Status.NOT_FOUND);  
    18.         }  
    19.         return new StreamingOutput() {  
    20.             public void write(OutputStream outputStream) throws IOException,  
    21.                     WebApplicationException {  
    22.                 outputCustomer(outputStream, customer);  
    23.             }  
    24.         };  
    25.     }  
    26.   
    27.     public void updateCustomer(int id, InputStream is) {  
    28.         Customer update = readCustomer(is);  
    29.         Customer current = customerDB.get(id);  
    30.         if (current == null)  
    31.             throw new WebApplicationException(Response.Status.NOT_FOUND);  
    32.         current.setFirstName(update.getFirstName());  
    33.         current.setLastName(update.getLastName());  
    34.         current.setStreet(update.getStreet());  
    35.         current.setState(update.getState());  
    36.         current.setZip(update.getZip());  
    37.         current.setCountry(update.getCountry());  
    38.     }  
    39.   
    40.     protected void outputCustomer(OutputStream os, Customer cust)  
    41.             throws IOException {  
    42.         PrintStream writer = new PrintStream(os);  
    43.         writer.println("<customer id="" + cust.getId() + "">");  
    44.         writer.println(" <first-name>" + cust.getFirstName() + "</first-name>");  
    45.         writer.println(" <last-name>" + cust.getLastName() + "</last-name>");  
    46.         writer.println(" <street>" + cust.getStreet() + "</street>");  
    47.         writer.println(" <city>" + cust.getCity() + "</city>");  
    48.         writer.println(" <state>" + cust.getState() + "</state>");  
    49.         writer.println(" <zip>" + cust.getZip() + "</zip>");  
    50.         writer.println(" <country>" + cust.getCountry() + "</country>");  
    51.         writer.println("</customer>");  
    52.     }  
    53.   
    54.     protected Customer readCustomer(InputStream is) {  
    55.         try {  
    56.             DocumentBuilder builder = DocumentBuilderFactory.newInstance()  
    57.                     .newDocumentBuilder();  
    58.             Document doc = builder.parse(is);  
    59.             Element root = doc.getDocumentElement();  
    60.             Customer cust = new Customer();  
    61.             if (root.getAttribute("id") != null  
    62.                     && !root.getAttribute("id").trim().equals("")) {  
    63.                 cust.setId(Integer.valueOf(root.getAttribute("id")));  
    64.             }  
    65.             NodeList nodes = root.getChildNodes();  
    66.             for (int i = 0; i < nodes.getLength(); i++) {  
    67.                 Node item = nodes.item(i);  
    68.                 if(!(item instanceof Element)){  
    69.                     continue;  
    70.                 }  
    71.                 Element element = (Element) nodes.item(i);  
    72.                 if (element.getTagName().equals("first-name")) {  
    73.                     cust.setFirstName(element.getTextContent());  
    74.                 } else if (element.getTagName().equals("last-name")) {  
    75.                     cust.setLastName(element.getTextContent());  
    76.                 } else if (element.getTagName().equals("street")) {  
    77.                     cust.setStreet(element.getTextContent());  
    78.                 } else if (element.getTagName().equals("city")) {  
    79.                     cust.setCity(element.getTextContent());  
    80.                 } else if (element.getTagName().equals("state")) {  
    81.                     cust.setState(element.getTextContent());  
    82.                 } else if (element.getTagName().equals("zip")) {  
    83.                     cust.setZip(element.getTextContent());  
    84.                 } else if (element.getTagName().equals("country")) {  
    85.                     cust.setCountry(element.getTextContent());  
    86.                 }  
    87.             }  
    88.             return cust;  
    89.         } catch (Exception e) {  
    90.             throw new WebApplicationException(e, Response.Status.BAD_REQUEST);  
    91.         }  
    92.     }  
    93. }  

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

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

    我喜欢程序员,他们单纯、固执、容易体会到成就感;面对压力,能够挑灯夜战不眠不休;面对困难,能够迎难而上挑战自我。他 们也会感到困惑与傍徨,但每个程序员的心中都有一个比尔盖茨或是乔布斯的梦想“用智慧开创属于自己的事业”。我想说的是,其 实我是一个程序员
  • 相关阅读:
    Python 学习笔记:需要仔细阅读一个函数
    asp.net 实现一个简单CAS Server
    用代码旋转屏幕
    Cassandra API60 Java 代码示例
    Ajax跨域访问代理类,支持GET和POST方法
    Android 给自己的类加个事件
    Python 学习笔记: 备份工具
    Android SQLiteHelper
    转:在 CLI 中練習 Data Model
    Java修饰符public,private,protected及默认的区别
  • 原文地址:https://www.cnblogs.com/kms1989/p/5822955.html
Copyright © 2011-2022 走看看