一篇文章帶你搞懂Java restful 接口開發

1、RESTful 簡介

REST:Representational State Transfer,表現層資源狀態轉移。

a>資源

資源是一種看待服務器的方式,即,將服務器看作是由很多離散的資源組成。每個資源是服務器上一個可命名的抽象概念。因為資源是一個抽象的概念,所以它不僅僅能代表服務器文件系統中的一個文件、數據庫中的一張表等等具體的東西,可以將資源設計的要多抽象有多抽象,隻要想象力允許而且客戶端應用開發者能夠理解。與面向對象設計類似,資源是以名詞為核心來組織的,首先關註的是名詞。一個 資源可以由一個或多個 URI 來標識。URI 既是資源的名稱,也是資源在 Web 上的地址。對某個資源感興趣的客戶端應用,可以通過資源的 URI 與其進行交互。

b>資源的表述

資源的表述是一段對於資源在某個特定時刻的狀態的描述。可以在客戶端-服務器端之間轉移(交換)。資源的表述可以有多種格式,例如 HTML/XML/JSON/純文本/圖片/視頻/音頻等等。資源的表述格式可以通過協商機制來確定。請求-響應方向的表述通常使用不同的格式。

c>狀態轉移

狀態轉移說的是:在客戶端和服務器端之間轉移(transfer)代表資源狀態的表述。通過轉移和操作資源的表述,來間接實現操作資源的目的。

2、RESTful 的實現

具體說,就是 HTTP 協議裡面,四個表示操作方式的動詞:GET、POST、PUT、DELETE。

它們分別對應四種基本操作:GET 用來獲取資源,POST 用來新建資源,PUT 用來更新資源,DELETE 用來刪除資源。

REST 風格提倡 URL 地址使用統一的風格設計,從前到後各個單詞使用斜杠分開,不使用問號鍵值對方式攜帶請求參數,而是將要發送給服務器的數據作為 URL 地址的一部分,以保證整體風格的一致性。

操作 傳統方式 REST 風格
查詢操作 getUserById?id=1 user/1–>get 請求方式
保存操作 saveUser user–>post 請求方式
刪除操作 deleteUser?id=1 user/1–>delete 請求方式
更新操作 updateUser user–>put 請求方式

3、HiddenHttpMethodFilter

由於瀏覽器隻支持發送 get 和 post 方式的請求,那麼該如何發送 put 和 delete 請求呢?

SpringMVC 提供瞭 HiddenHttpMethodFilter 幫助我們將 POST 請求轉換為 DELETE 或 PUT 請求。

HiddenHttpMethodFilter 處理 put 和 delete 請求的條件:

  • 當前請求的請求方式必須為 post
  • 前請求必須傳輸請求參數_method

滿足以上條件,HiddenHttpMethodFilter 過濾器就會將當前請求的請求方式轉換為請求參數_method 的值,因此請求參數_method 的值才是最終的請求方式。

在 web.xml 中註冊 HiddenHttpMethodFilter

<filter>
    <filter-name>HiddenHttpMethodFilter</filter-name>
    <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filterclass>
</filter>
<filter-mapping>
    <filter-name>HiddenHttpMethodFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

使用註意:

目前為止,SpringMVC 中提供瞭兩個過濾器:CharacterEncodingFilter 和 HiddenHttpMethodFilter

在 web.xml 中註冊時,必須先註冊 CharacterEncodingFilter,再註冊 HiddenHttpMethodFilter

原因:

在 CharacterEncodingFilter 中通過 request.setCharacterEncoding(encoding) 方法設置字符集的

request.setCharacterEncoding(encoding) 方法要求前面不能有任何獲取請求參數的操作而 HiddenHttpMethodFilter 恰恰有一個獲取請求方式的操作:

String paramValue = request.getParameter(this.methodParam);

4、RESTful 案例

4.1、需求

通過 restfull 實現用戶的增刪改查,需要提供 5 個接口。

接口 method 描述
/user/list GET 獲取用戶列表
/user/{userId} GET 根據用戶 id 獲取用戶信息
/user POST 新增用戶信息
/user PUT 保存用戶信息
/user/{userId} DELETE 刪除用戶信息

4.2、git 代碼位置

https://gitee.com/javacode2018/springmvc-series

9c531dfb77b5870eb83558aff4c15c0d.png

4.3、UserController

UserController 中實現瞭需求中提到的 4 個接口,大傢重點看下每個接口的方法上用到的註解。

@RestController
public class UserController {
    private List<User> userList = new ArrayList<>();
 
    {
        userList.add(new User(1, "Spring高手系列"));
        userList.add(new User(2, "SpringMVC系列"));
    }
 
    @GetMapping("/user/list")
    public List<User> list() {
        System.out.println("list()");
        return userList;
    }
 
    @GetMapping("/user/{userId}")
    public User getUser(@PathVariable("userId") Integer userId) {
        System.out.println("getUser()");
        for (User user : userList) {
            if (user.getUserId().equals(userId)) {
                return user;
            }
        }
        return null;
    }
 
    @PostMapping(value = "/user", produces = "text/html;charset=UTF-8")
    public String add(User user) {
        System.out.println("add()");
        this.userList.add(user);
        return "新增成功";
    }
 
    @PutMapping(value = "/user", produces = "text/html;charset=UTF-8")
    public String modify(User user) {
        System.out.println("modify()");
        for (User item : userList) {
            if (item.getUserId().equals(user.getUserId())) {
                item.setName(user.getName());
            }
        }
        return "修改成功";
    }
 
    @DeleteMapping(value = "/user/{userId}", produces = "text/html;charset=UTF-8")
    public String delete(@PathVariable("userId") Integer userId) {
        System.out.println("delete()");
        Iterator<User> iterator = userList.iterator();
        while (iterator.hasNext()) {
            User user = iterator.next();
            if (user.getUserId().equals(userId)) {
                iterator.remove();
            }
        }
        return "刪除成功";
    }
 
 
    public static class User {
        private Integer userId;
        private String name;
 
        public User() {
        }
 
        public User(Integer userId, String name) {
            this.userId = userId;
            this.name = name;
        }
 
        public Integer getUserId() {
            return userId;
        }
 
        public void setUserId(Integer userId) {
            this.userId = userId;
        }
 
        public String getName() {
            return name;
        }
 
        public void setName(String name) {
            this.name = name;
        }
 
        @Override
        public String toString() {
            return "User{" +
                    "userId=" + userId +
                    ", name='" + name + '\'' +
                    '}';
        }
    }
}

4.4、添加 HiddenHttpMethodFilter

HiddenHttpMethodFilter hiddenHttpMethodFilter = new HiddenHttpMethodFilter();

fb359e3e1eb468807e691156956ae084.png

4.5、測試效果

將項目發佈到 tomcat,如下圖,跑一下 UserController.http 中的 5 個用例,點擊每個用戶中的綠色箭頭即可運行,註意下後面 3 個用例都是 POST 方式提交的,但是參數中多瞭一個_method 參數用來指定提交的類型,這個參數會被HiddenHttpMethodFilter 解析。

5d2af105258372e24a7bea0f7b640c93.png

總結

本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!

推薦閱讀: