簡介
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服務用於對客戶進行管理,包括:
- 建立客戶
- 檢視客戶
- 更新客戶
首先給出對應的於這些操作的服務介面:
- 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服務的定義和實現分開,這樣程式碼的結構簡潔、清晰,在後期也可以很方便的進行實現的替換和服務定義的修改。
下面就是新增實現部分:
- 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定義都需要重寫,而不能僅修改變化的。