JAX-RS(java restful实现讲解)(转)

2024-03-16 10:18
文章标签 java rs 实现 讲解 restful jax

本文主要是介绍JAX-RS(java restful实现讲解)(转),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

JAX-RS入门 一 :基础

  • 博客分类: 
  • JAX-RS

简介

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代码   收藏代码
  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定义都需要重写,而不能仅修改变化的。

这篇关于JAX-RS(java restful实现讲解)(转)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/815154

相关文章

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Spring WebFlux 与 WebClient 使用指南及最佳实践

《SpringWebFlux与WebClient使用指南及最佳实践》WebClient是SpringWebFlux模块提供的非阻塞、响应式HTTP客户端,基于ProjectReactor实现,... 目录Spring WebFlux 与 WebClient 使用指南1. WebClient 概述2. 核心依

Mysql实现范围分区表(新增、删除、重组、查看)

《Mysql实现范围分区表(新增、删除、重组、查看)》MySQL分区表的四种类型(范围、哈希、列表、键值),主要介绍了范围分区的创建、查询、添加、删除及重组织操作,具有一定的参考价值,感兴趣的可以了解... 目录一、mysql分区表分类二、范围分区(Range Partitioning1、新建分区表:2、分

MySQL 定时新增分区的实现示例

《MySQL定时新增分区的实现示例》本文主要介绍了通过存储过程和定时任务实现MySQL分区的自动创建,解决大数据量下手动维护的繁琐问题,具有一定的参考价值,感兴趣的可以了解一下... mysql创建好分区之后,有时候会需要自动创建分区。比如,一些表数据量非常大,有些数据是热点数据,按照日期分区MululbU

Spring Boot @RestControllerAdvice全局异常处理最佳实践

《SpringBoot@RestControllerAdvice全局异常处理最佳实践》本文详解SpringBoot中通过@RestControllerAdvice实现全局异常处理,强调代码复用、统... 目录前言一、为什么要使用全局异常处理?二、核心注解解析1. @RestControllerAdvice2

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

MySQL中查找重复值的实现

《MySQL中查找重复值的实现》查找重复值是一项常见需求,比如在数据清理、数据分析、数据质量检查等场景下,我们常常需要找出表中某列或多列的重复值,具有一定的参考价值,感兴趣的可以了解一下... 目录技术背景实现步骤方法一:使用GROUP BY和HAVING子句方法二:仅返回重复值方法三:返回完整记录方法四:

IDEA中新建/切换Git分支的实现步骤

《IDEA中新建/切换Git分支的实现步骤》本文主要介绍了IDEA中新建/切换Git分支的实现步骤,通过菜单创建新分支并选择是否切换,创建后在Git详情或右键Checkout中切换分支,感兴趣的可以了... 前提:项目已被Git托管1、点击上方栏Git->NewBrancjsh...2、输入新的分支的

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.