【SpringMVC】3—RESTFul风格

⭐⭐⭐⭐⭐⭐
Github主页👉https://github.com/A-BigTree
笔记链接👉https://github.com/A-BigTree/Code_Learning
⭐⭐⭐⭐⭐⭐

如果可以,麻烦各位看官顺手点个star~😊

如果文章对你有所帮助,可以点赞👍收藏⭐支持一下博主~😆


3 RESTFul风格

3.1 RESTFul概述

3.1.1 REST概念

REST:Representational State Transfer,表现层资源状态转移。

  • 定位:互联网软件架构风格;
  • 倡导者:Roy Thomas Fielding;
  • 文献:Roy Thomas Fielding的博士论文;

3.1.2 REST规范的内涵

资源

URL:Uniform Resource Locator统一资源定位器。意思是网络上的任何资源都可以通过 URL 来定位。但是在实际开发中,我们往往是使用URL来对应一个具体的功能,而不是资源本身。REST规范则倡导使用URL对应网络上的各种资源,任何一个资源都可以通过一个URL访问到,为实现操作幂等性奠定基础。而这个资源可以是网络上的一个文本、音频、视频、图片等等……

幂等性:如果一个操作执行一次和执行N次对系统的影响相同,那么我们就说这个操作满足幂等性。而幂等性正是REST规范所倡导的。

状态转移

REST倡导针对资源本身操作,所以对资源的操作如果满足幂等性,那么操作只会导致资源本身的状态发生变化而不会破坏整个系统数据。

3.1.3 REST规范具体要求

四种请求方式

REST风格主张在项目设计、开发过程中,具体的操作符合HTTP协议定义的请求方式的语义

操作请求方式
查询操作GET
保存操作POST
删除操作DELETE
更新操作PUT

另有一种说法:

  • POST 操作针对功能执行,没有锁定资源 id,是非幂等性操作;
  • PUT 操作锁定资源 id,即使操作失败仍然可以针对原 id 重新执行,对整个系统来说满足幂等性:
    • id对应的资源不存在:执行保存操作
    • id对应的资源存在:执行更新操作
URL地址风格

REST风格提倡URL地址使用统一的风格设计,从前到后各个单词使用斜杠分开不使用问号键值对方式携带请求参数,而是将要发送给服务器的数据作为URL地址的一部分,以保证整体风格的一致性。还有一点是不要使用请求扩展名

传统 URL 地址REST 风格地址
/remove/emp?id=5/emp/5
  • 用一句话描述当前资源
  • 一句话中各个单词用斜杠分开,从前到后保持完全一致的书写风格
  • 不要使用问号键值对的方式传递数据
  • 需要传递数据时,把数据嵌入到URL地址中,作为地址的一部分
  • 不要使用请求扩展名

3.1.4 REST风格的好处

含蓄安全

使用问号键值对的方式给服务器传递数据太明显,容易被人利用来对系统进行破坏。使用 REST 风格携带数据不再需要明显的暴露数据的名称。

风格统一

URL 地址整体格式统一,从前到后始终都使用斜杠划分各个单词,用简单一致的格式表达语义。

无状态

在调用一个接口(访问、操作资源)的时候,可以不用考虑上下文,不用考虑当前状态,极大的降低了系统设计的复杂度。

严谨优雅

严格按照HTTP1.1协议中定义的请求方式本身的语义进行操作。

简洁优雅

过去做增删改查操作需要设计4个不同的URL,现在一个就够了。

操作传统风格REST 风格
保存/CRUD/saveEmpURL 地址:/CRUD/emp
请求方式:POST
删除/CRUD/removeEmp?empId=2URL 地址:/CRUD/emp/2
请求方式:DELETE
更新/CRUD/updateEmpURL 地址:/CRUD/emp
请求方式:PUT
查询(表单回显)/CRUD/editEmp?empId=2URL 地址:/CRUD/emp/2
请求方式:GET
丰富的语义

通过 URL 地址就可以知道资源之间的关系。它能够把一句话中的很多单词用斜杠连起来,反过来说就是可以在 URL 地址中用一句话来充分表达语义。

3.2 四种请求方式映射

3.2.1 HiddenHttpMethodFilter与装饰模式

简介

在HTML中,GET和POST请求可以天然实现,但是DELETE和PUT请求无法直接做到。SpringMVC提供了HiddenHttpMethodFilter帮助我们将POST请求转换为DELETE或PUT请求。

HiddenHttpMethodFilter要点

默认请求参数名常量:

public static final String DEFAULT_METHOD_PARAM = "_method";

HiddenHttpMethodFilter中,声明了一个常量:DEFAULT_METHOD_PARAM,常量值是"_method"

配套的成员变量:

private String methodParam = DEFAULT_METHOD_PARAM;

之所以会提供这个成员变量和配套的setXxx()方法,是允许我们在配置Filter时,通过初始化参数来修改这个变量。如果不修改,默认就是前面常量定义的值。

在这里插入图片描述

原始请求对象的包装

困难:

  • 包装对象必须和原始对象是同一个类型
  • 保证同一个类型不能通过子类继承父类实现
    • 子类对象:希望改变行为、属性的对象
    • 父类对象:随着Servlet容器的不同,各个容器对HttpServletRequest接口给出的实现不同。如果继承了 A 容器给出的实现类,那么将来就不能再迁移到 B 容器。
  • 只能让包装对象和被包装对象实现相同接口
    • 虽然使用动态代理技术大致上应该能实现,但是一旦应用代理就必须为被包装的对象的每一个方法都进行代理,操作过于繁琐。
  • 如果我们自己创建一个类实现HttpServletRequest接口
    • 困难1:在不确定具体哪一个 Servlet 容器的情况下完全没办法实现
    • 困难2:抽象方法实在太多

HttpServletRequestWrapper

HttpServletRequestWrapper类能够非常好的帮助我们对原始request对象进行包装。它为什么能帮我们解决上面的困难呢?

  • HttpServletRequestWrapper 类替我们实现了HttpServletRequest接口;
  • 为了让包装得到的新对象在任何Servlet容器平台上都能够正常工作,HttpServletRequestWrapper类此处的设计非常巧妙:它借助原始的request对象本身来实现所有的具体功能;
  • 在我们想通过包装的方式来修改原始对象的行为或属性时,只需要在HttpServletRequestWrapper 类的子类中重写对应的方法即可;

HttpMethodRequestWrapper类:

HttpMethodRequestWrapper类就是HiddenHttpMethodFilter 的一个内部类,在HttpMethodRequestWrapper类中有如下行为实现了对原始对象的包装:

  • 继承了官方包装类:HttpServletRequestWrapper
  • 在构造器中将原始request对象传给了父类构造器;
  • 将我们指定的新请求方式传给了成员变量;
  • 重写了父类(官方包装类)的getMethod()方法;
  • 外界想知道新包装对象的请求方式时,会来调用被重写的getMethod()方法,从而得到我们指定的请求方式;
/**
 * Simple {@link HttpServletRequest} wrapper that returns the supplied method for
 * {@link HttpServletRequest#getMethod()}.
 */
private static class HttpMethodRequestWrapper extends HttpServletRequestWrapper {
  
  private final String method;
  
  public HttpMethodRequestWrapper(HttpServletRequest request, String method) {
        // 在构造器中将原始 request 对象传给了父类构造器
    super(request);
        
        // 将我们指定的新请求方式传给了成员变量
    this.method = method;
  }
  
  @Override
  public String getMethod() {
    return this.method;
  }
}

3.2.2 PUT请求

web.xml
<filter>
    <filter-name>hiddenHttpMethodFilter</filter-name>
    <filter-class>org.springframework.web.filter.HiddenHttpMethodFilter</filter-class>
</filter>
<filter-mapping>
    <filter-name>hiddenHttpMethodFilter</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
表单
  • 要点1:原请求方式必须是 post
  • 要点2:新的请求方式名称通过请求参数发送
  • 要点3:请求参数名称必须是_method
  • 要点4:请求参数的值就是要改成的请求方式
<!-- 原请求方式必须是 post -->
<form th:action="@{/emp}" method="post">
    <!-- 通过表单隐藏域携带一个请求参数 -->
    <!-- 请求参数名:_method -->
    <!-- 请求参数值:put -->
    <input type="hidden" name="_method" value="put" />

    <button type="submit">更新</button>
</form>
处理方法
// 映射请求地址:URL + 请求方式
@RequestMapping(value = "/emp", method = RequestMethod.PUT)
public String updateEmp() {
    
    logger.debug("现在执行的是 updateEmp() 方法");
    
    return "target";
}

3.2.3 DELETE请求

场景
<h3>将XXX请求转换为DELETE请求</h3>
<table id="dataTable">
    <tr>
        <th>姓名</th>
        <th>年龄</th>
        <th>删除</th>
    </tr>
    <tr>
        <td>张三</td>
        <td>40</td>
        <td>
            <a th:href="@{/emp}" @click="doConvert">删除</a>
        </td>
    </tr>
    <tr>
        <td>李四</td>
        <td>30</td>
        <td>
            <a th:href="@{/emp}" @click="doConvert">删除</a>
        </td>
    </tr>
</table>
负责转换的表单
<!-- 创建一个通用表单,在删除超链接的单击响应函数中通过这个表单把GET请求转换为POST,进而再转DELETE -->
<form method="post" id="convertForm">
    <input type="hidden" name="_method" value="delete" />
</form>
删除超链接绑定单击响应函数
<td>
    <!-- /emp/{empId}/{pageNo} -->
    <!-- οnclick="convertMethod(this)" 表示点击这个超链接时,调用 convertMethod() 函数 -->
    <!-- this 代表当前超链接对象 -->
    <!-- event 是代表当前事件的事件对象 -->
    <a onclick="convertMethod(this, event)" th:href="@{/emp}">删除</a>
</td>
编写单击响应函数
<script type="text/javascript">

    function convertMethod(anchorElement, event) {

        // 获取超链接原本要访问的目标地址
        var targetURL = anchorElement.href;

        // 获取表单对象
        var formEle = document.getElementById("convertForm");

        // 把超链接原本要访问的地址设置给表单的 action 属性
        formEle.action = targetURL;

        // 提交表单
        formEle.submit();

        // 取消控件的默认行为:让超链接不会跳转
        event.preventDefault();
    }

</script>
处理方法
@RequestMapping(value = "/emp", method = RequestMethod.DELETE)
public String removeEmp() {
    
    logger.debug("现在执行的是 removeEmp() 方法");
    
    return "target";
}

3.3 @PathVariable注解

3.3.1 操作

传一个值
<a th:href="@{/emp/20}">传一个值</a><br/>
// 实际访问地址:/emp/20
// 映射地址:/emp/{empId}是把变量部分用大括号标记出来,写入变量名
@RequestMapping("/emp/{empId}")
public String getEmpById(@PathVariable("empId") Integer empId) {
    
    logger.debug("empId = " + empId);
    
    return "target";
}
传多个值
<a th:href="@{/emp/tom/18/50}">传多个值</a><br/>
处理方法
// 实际地址:/emp/tom/18/50
@RequestMapping("/emp/{empName}/{empAge}/{empSalary}")
public String queryEmp(
        @PathVariable("empName") String empName,
        @PathVariable("empAge") Integer empAge,
        @PathVariable("empSalary") Double empSalary
) {
    
    logger.debug("empName = " + empName);
    logger.debug("empAge = " + empAge);
    logger.debug("empSalary = " + empSalary);
    
    return "target";
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一棵___大树

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值