SpringMVC 進階

来源:https://www.cnblogs.com/yangyuanhu/archive/2020/02/13/12304786.html

請求限制 一些情況下我們可能需要對請求進行限制,比如僅允許POST,GET等... RequestMapping註解中提供了多個參數用於添加請求的限制條件 value 請求地址 path 請求地址 method 請求方法 headers 請求頭中必須包含指定欄位 params 必須包含某個請求參數 ...


請求限制

一些情況下我們可能需要對請求進行限制,比如僅允許POST,GET等...

RequestMapping註解中提供了多個參數用於添加請求的限制條件

  • value 請求地址
  • path 請求地址
  • method 請求方法
  • headers 請求頭中必須包含指定欄位
  • params 必須包含某個請求參數
  • consumes 接受的數據媒體類型 (與請求中的contentType匹配才處理)
  • produce 返回的媒體類型 (與請求中的accept匹配才處理)

案例:

@RequestMapping(value = "/editCourse",method = RequestMethod.POST,headers = {"id"},params = {"name"},consumes = {"text/plain"},produces = {"text/html"})
//含義:url為/editCourse 請求方法為POST hander必須包含id欄位  參數必須包含name 只接受text/plain類型數據 返回數據類型為text/html

為了簡化書寫,MVC還提供了集合路徑和方法限制的註解,包括常見的請求方法:

PostMapping
GetMapping
DeleteMapping
PutMapping

例:
@PostMapping("/editCourse")

handler返回值

handler方法可以是三種類型的返回值,用於不同場景

ModelAndView

返回值為視圖和數據的包裝類型,用於返回邏輯視圖名稱和視圖需要展示的數據

等同於在Request中添加了屬性,然後進行了請求轉發

例:

@RequestMapping("/test")
public ModelAndView test() {
    ModelAndView modelAndView = new ModelAndView();
    modelAndView.setViewName("index.jsp");
    modelAndView.addObject("msg", "hello ssm!");
    return modelAndView;
}

void

表示handler不返回任何數據,用於當需要直接操作response完成響應的場景

例:

@RequestMapping("/test2")
public void test2(String name,HttpServletResponse response) throws IOException {
    response.getWriter().println(name.toUpperCase());
}

String

返回一個字元串類型的值,返回的內容可以是視圖名稱也可以是其他請求地址

其背後採用的是請求轉發的方式

例:

@RequestMapping("/test3")
public String test3(Model model) {
    model.addAttribute("msg","hello XXX");
    return "index.jsp";
}
@RequestMapping("/test4")
public String test4() {
    return "/test";
}

轉發和重定向:

也可指定對目標地址的請求是通過重定向或請求轉發;

例:

@RequestMapping("/test5")
public String test5() {
    return "forward:/index.jsp";
}
@RequestMapping("/test6")
public String test6() {
    return "redirect:/index.jsp";
}

當然了 預設就是forward所以可以省略;

json交互

當下,大多數公司都會有移動端App,當我們的後臺服務需要為App提供介面時,就不得不使用到json數據了,當然還有前後端分離項目中前端和後臺同樣採用json來交換數據;

在開始前,需要導入jackson依賴,用於實現json的序列化與反序列化;

<dependency>
  <groupId>com.fasterxml.jackson.core</groupId>
  <artifactId>jackson-databind</artifactId>
  <version>2.9.9</version>
</dependency>

返回json數據

@ResponseBody註解用於標註一個handler方法返回的是json數據,同時方法的返回值將作為返回給前臺的數據;

例:

@RequestMapping("/getCourseList")
@ResponseBody
public List<Course> getCourseList() {  //獲取所有課程
    return courseService.selectCourses();
}
@RequestMapping("/getCourse")
@ResponseBody
public Course getCourse(Integer id) {// 根據id獲取一個課程
    return courseService.selectByID(id);
}

ResponseBody會將響應的ContentType設置為application/json, 然後調用jackson的toJsonString將返回值轉為json字元串,最後返回給客戶端;

@RestController

如果需要為每一個方法添加ResponseBody的話,就顯得非常麻煩,SpringMVC提供了@RestController註解,表示這是一個所有handler返回值全都是json的Controller,相當於把Controller和ResponseBody兩個註解合併在一起;

例:

@RestController
public class CourseController {.....}

接受json數據

SpringMVC可以幫助我們將json參數反序列化到指定的實體類型,List或Map;

需要強調的是:客戶端必須指定ContenType為application/json

@RequestMapping("/addCourse")
@ResponseBody
public Course addCourse(@RequestBody Course course) {//接收json參數映射到實體
    course.setName("接收json成功");//修改name再把數據發回去 以便查看效果
    return course;
}

@RequestMapping("/addCourses")
@ResponseBody
public List<Course> addCourse(@RequestBody List<Course> courses) {//接收json數組參數映射到list
    return courses;
}

@RequestMapping("/addData")
@ResponseBody
public Map<String,String> addData(@RequestBody Map<String,String> data) {//接收json數據映射到map
    return data;
}

@RequestMapping("/addInfo")
@ResponseBody
public String addInfo(@RequestBody String data) {//接收json數據不做任何轉換
    return data;
}

當客戶端傳遞的json比較複雜時可能無法直接轉換到某個實體類型,這是我們可以通過Map來接收,或直接獲取原始的json字元串自己處理; 就像上面的addDataaddInfo一樣

Handler攔截器

顧名思義Handler攔截器可對Handler方法進行攔截,控制Handler方法是否執行,與Servlet的過濾器非常相似

但是要註意:

​ Servlet的filter的執行時機是在SpringMVC之前,過濾的目標對象是請求;

​ 而Handler攔截器,攔截的目標對象是Handler方法

Handler攔截器可以方便的實現,登錄狀態驗證,操作許可權驗證等操作;

使用案例:

1.編寫攔截器

public class MyInterceptor implements HandlerInterceptor {
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        //在執行handler前調用   返回值將決定是否繼續執行請求
        System.out.println("preHandle");
        return true;
    }

    @Override
    public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
        //handler被真正執行了,已經拿到了handler的返回結果  但是DispatcherServlet還沒有發送給前臺
        System.out.println("postHandle");
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        //DispatcherServlet將視圖發送給前臺後的回調 (無論handler是否執行 一定有響應發給前臺)
        System.out.println("afterCompletion");
      
        //Handler中出現的任何異常也會傳給該方法,可以在這裡進行處理
          if (ex != null){
            System.out.println("handler中出現異常了....");
        }
    }
}

2.配置攔截器

<mvc:interceptors>
    <mvc:interceptor>
        <mvc:mapping path="/**"/>
        <bean class="com.kkb.interceptor.MyInterceptor"/>
    </mvc:interceptor>
</mvc:interceptors>
<!--
* path用於指定攔截器攔截的url只有handler的url與之匹配才會被攔截  /** 表示攔截所有請求
* interceptors中可配置多個interceptor 
-->

3.執行順序

攔截器的執行順序由配置順序來決定,攔截器也和filter一樣是一個鏈條的形式

在請求處理完成時,會按照相反的順序通知interceptor(即執行afterComplation),前提是這個攔截器正常放行了請求(preHandler中返回了true),否則不會收到通知;

異常處理

一個完整的系統必然要考慮異常情況的處理,SpringMVC提供了一種非常方便的處理方法,只需要實現HandlerExceptionResolver介面,並註冊Bean至容器中即可

1.編寫異常處理器

@Component
public class MyExceptionHandler implements HandlerExceptionResolver {
    @Override
    public ModelAndView resolveException(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) {
        ModelAndView modelAndView = new ModelAndView();
        modelAndView.setViewName("error.jsp");
        modelAndView.addObject("exobj",ex);
        return modelAndView;
    }
}

可以看到在處理方法的返回值為ModelAndView,我們需要在其中添加錯誤頁面的名稱和錯誤信息;

2.註冊到容器中

​ 可以直接添加Component註解或是,在配置文件中註冊

<bean class="com.kkb.exceptionhandler.MyExceptionHandler"/>

強調:無論是攔截器還是異常處理器都是針對handler而不是所有請求,舉個例子如果請求本身就是錯誤的如404,是無法被異常Handler異常處理器處理的,仍需要到web.xml來進行配置


您的分享是我們最大的動力!

更多相關文章
  • JavaSE學習筆記(12) 線程 多線程 併發與並行 併發 :指兩個或多個事件在 同一個時間段內 發生。 並行 :指兩個或多個事件在 同一時刻 發生(同時發生)。 在操作系統中,安裝了多個程式,併發指的是在一段時間內巨集觀上有多個程式同時運行,這在單 CPU 系統中,每一時刻只能有一道程式執行,即微 ...
  • MySQL官方對索引的定義為:索引(Index)是幫助MySQL高效獲取數據的數據結構。 資料庫查詢是資料庫的最主要功能之一,我們都希望查詢數據的速度能儘可能的快,因此資料庫系統的設計者會從查詢演算法的角度進行優化,這篇文章對索引做一個系統的梳理,希望對大家有幫助。 一、MySQL有哪些索引類型 索引 ...
  • 最近需要設計一個API伺服器,想要把API介面搞得規範一下,就通過網上搜集到了一些資料,以下便是自己的一些理解以及相關的具體實現 本文采用的是spring boot+maven的方案 restful規範 這個規範我在這裡也不打算長篇大論地講解,怎麼說呢,有人喜歡有人討厭,我也不去爭,因為我經驗不多, ...
  • 一、判斷一個數字X的i位是不是1 二、把一個數字二進位下的第i位改成1 三、把一個數字二進位下的最靠右的第一個1改成0(去掉) ...
  • windows破解教程 1. 首先下載jar包:(雲盤鏈接發不上去,大家關註gzh" 灰太狼學爪哇 "回覆 idea 獲取)將其放到合適的文件夾(首選IDEA的同級目錄)進行管理; 2. 進入C盤 — 用戶 — 用戶名 — .IntelliJIdea2019.2或者.IntelliJIdea2019 ...
  • VisualVM在Java 8中是JDK自帶的一個圖形化工具,項目主頁 "VisualVM" ,在後續版本中可能會從JDK移除。 VisualVM可以監控Java進程的CPU與記憶體占用情況,可以監控Java進程內的各個線程的執行情況,還可以與MAT工具一樣用來分析堆轉儲快照。 監控遠程Tomcat進 ...
  • 1.前提條件 1). 確保已經安裝需要的Python版本 2). 確保已經將Python的目錄加入到環境變數中 2. Python安裝包的幾種常用方式 1). pip安裝方式(正常線上安裝) 2). whl安裝方式(離線安裝),一般是.whl格式的包 3). 源碼安裝方式(離線安裝),tar.gz/ ...
  • 概念: 什麼是REST? REST是Representational State Transfer的縮寫。翻譯為"表現層狀態轉化",restful是一種介面設計風格,它不是一個協議,通常是基於HTTP協議的; 為什麼需要這麼一個風格呢? RESTful的重點之一就是統一的介面命名規則; 每個開發者可 ...
一周排行
  • 在園子裡面有很多關於各種技術細節的研究文章,都是比較牛逼的框架研究;但是一直沒有看到關於怎麼樣提高開發效率的文章,大多提高開發效率的文章都是關於自動化等方面的輔助工具類型的,而不是開發中的一些小技巧;今天從編碼規範、編碼技巧、開發思想、設計模式等各方面的經驗來分享如何提高開發效率。 ...
  • 前言 隨著近些年微服務的流行,有越來越多的開發者和團隊所採納和使用,它的確提供了很多的優勢也解決了很多的問題,但是我們也知道也並不是銀彈,提供優勢的同時它也給我們的開發人員和團隊也帶來了很多的挑戰。 為了迎接或者採用這些新技術,開發團隊需要更加註重一些流程或工具的使用,這樣才能更好的適應這些新技術所 ...
  • 本文是本系列的完結篇。本系列前面的文章: 邏輯式編程語言極簡實現(使用C#) - 1. 邏輯式編程語言介紹 邏輯式編程語言極簡實現(使用C#) - 2. 一道邏輯題:誰是凶手 邏輯式編程語言極簡實現(使用C#) - 3. 運行原理 下午,吃飽飯的老明和小皮,各拿著一杯剛買的咖啡回到會議室,開始了邏輯 ...
  • 微服務之間的通信之gRPC 介紹 gRPC是一種與語言無關的高性能遠程過程調用 (RPC) 框架,gRPC是Google發佈的基於HTTP 2.0傳輸層協議承載的高性能開源軟體框架,提供了支持多種編程語言的、對網路設備進行配置和納管的方法。由於是開源框架,通信的雙方可以進行二次開發,所以客戶端和服務 ...
  • 一、TLS 線程本地存儲(Thread Local Storage),字面意思就是專屬某個線程的存儲空間。變數大體上分為全局變數和局部變數,一個進程中的所有線程共用地址空間,這個地址空間被劃分為幾個固有的區域,比如堆棧區,全局變數區等,全局變數存儲在全局變數區,虛擬地址固定;局部變數存儲在堆棧區,虛... ...
  • private:私有成員,在類的內部才可以訪問。 protected:保護成員,該類內部和繼承類中可以訪問。 public:公共成員,完全公開,沒有訪問限制。 internal:當前程式集內可以訪問。 ...
  • 前言 上一篇【.Net Core微服務入門全紀錄(六)——EventBus-事件匯流排】中使用CAP完成了一個簡單的Eventbus,實現了服務之間的解耦和非同步調用,並且做到數據的最終一致性。這一篇將使用IdentityServer4來搭建一個鑒權中心,來完成授權認證相關的功能。 IdentitySe ...
  • using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System. ...
  • 從今天起,我將製作一個電影推薦項目,在此寫下博客,記錄每天的成果。 其實,從我發佈 C# 爬取貓眼電影數據 這篇博客後, 我就已經開始製作電影推薦項目了,今天寫下這篇博客,也是因為項目進度已經完成50%了,我就想在這一階段停一下,回顧之前學到的知識。 一、主要為手機端 考慮到項目要有實用性,我選擇了 ...
  • 一、實現Runnable介面 public class RunnableDemo implements Runnable { public void run() { try { Thread.sleep(100); } catch (InterruptedException e) { e.print ...