Spring系列 SpringMVC的請求與資料響應

一隻胡說八道的猴子發表於2020-10-15

Spring系列 SpringMVC的請求與資料響應

在這裡插入圖片描述

SpringMVC的資料響應

資料響應的方式

y以下案例均部署在Tomcat上,使用瀏覽器來訪問一個簡單的success.jsp頁面來實現

Success.jsp頁面程式碼

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<H1>Success</H1>
</body>
</html>

1.頁面跳轉
直接返回字串
返回Model與View模型

2.回寫資料
直接返回字串
返回物件或集合

在spring-mvc.xml中配置內部檢視資源解析器

<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
        <property name="prefix" value="/web/WEB-INF/jsp/"></property>
        <property name="suffix" value=".jsp"></property></bean>

1.頁面跳轉

1.1 直接返回字串

這個比較簡單

@Controller
public class Usercontroller {
    @RequestMapping(value = "/quick2")
    public String save2(){
        System.out.println("Controller save running!!");
        return "/success.jsp";
    }
}

結果

在這裡插入圖片描述

1.2返回Model與View模型

方法一

@Controller
public class Usercontroller {
   @RequestMapping(value = "/quick3")
    public ModelAndView save3(){
        /*建立Modelandview物件*/
        /*
        * model模型用於封裝資料
        * view模型用於展示資料
        * */
        ModelAndView modelAndView = new ModelAndView();
        /*進行資料封裝*/
        modelAndView.addObject("Data","This is data" );
       /*設定檢視名稱*/
        modelAndView.setViewName("success");
        /*直接返回model and View*/
        return modelAndView;
    }
    
}

success.jsp頁面程式碼

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<%--獲取model封裝的資料--%>
<H1>Success ${Data}</H1>
</body>
</html>

頁面效果
在這裡插入圖片描述

方法二
由Spring框架來給我們自動注入

這個方法與前一個方法的區別就是不用我們自己來建立ModelandView物件,Spring框架會給我們自動注入

@Controller
public class Usercontroller {
   @RequestMapping(value = "/quick4")
    public ModelAndView save4(ModelAndView modelAndView){

        /*進行資料封裝*/
        modelAndView.addObject("Data","This is data" );
        /*設定檢視名稱*/
        modelAndView.setViewName("success");
        /*直接返回model and View*/
        return modelAndView;
    }
   
}

方法三
直接使用原生的HttpServletRequest物件

@Controller
public class Usercontroller {
  
   @RequestMapping(value = "/quick5")
    public String save4(HttpServletRequest request){
        /*進行資料封裝*/
       request.setAttribute("Data","This is data");
        return "success";
    }
}

1.3回寫字串不進行跳轉

方法一:通過SpringMVC框架注入的response物件,使用response.getWriter().print(“hello world”) 回寫資料,此時不需要檢視跳轉,

@Controller
public class Usercontroller {
  @RequestMapping(value = "/quick6")
    public void save6(HttpServletResponse response) throws IOException {
        response.getWriter().println("This is save write");
    }
}

方法二:將需要回寫的字串直接返回,但此時需要通過@ResponseBody註解告知SpringMVC框架,方法返回的字串不是跳轉是直接在http響應體中返回

@Controller
public class Usercontroller {
  @RequestMapping(value = "/quick6")
    /*告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save6(HttpServletResponse response) throws IOException {
       return  "This is save write";
    }
}

結果:

在這裡插入圖片描述
4.使用json格式回寫字串
手動拼接json格式字串的方式很麻煩,開發中往往要將複雜的java物件轉換成json格式的字串,我們可以使用web階段學習過的json轉換工具jackson進行轉換,通過jackson轉換json格式字串,回寫字串

@Controller
public class Usercontroller {
   @RequestMapping(value = "/quick7")
   /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public String save7() throws IOException {
        User user = new User();
        user.setAge(11);
        user.setName("Lisi");
        //使用jason轉換工具將物件轉換為json格式
        ObjectMapper objectMapper = new ObjectMapper();
        String json = objectMapper.writeValueAsString(user);
        return json;
    }
}

5.SpringMVC的資料響應-回寫資料-返回物件或集合

使用配置檔案方式
在spring-mvc.XML中經行如下配置

<bean class="org.springframework.web.servlet.mvc.method.annotation
             .RequestMappingHandlerAdapter"> 
                <property name="messageConverters">       
                 <list>           
                  <bean class="org.springframework.http.converter.json
                       .MappingJackson2HttpMessageConverter">
            </bean>    
                </list>  
                  </property>
                  </bean>
@RequestMapping("/quick8")
@ResponseBodypublic User quickMethod8() throws IOException
 {   
  User user = new User();  
   user.setUsername("zhangsan");   
    user.setAge(18);    
    return user;}

使用註解方式
在方法上新增@ResponseBody就可以返回json格式的字串,但是這樣配置比較麻煩,配置的程式碼比較多,因此,我們可以使用mvc的註解驅動代替上述配置。

在 SpringMVC 的各個元件中,處理器對映器、處理器介面卡、檢視解析器稱為 SpringMVC 的三大元件。
使用mvc:annotation-driven自動載入 RequestMappingHandlerMapping(處理對映器)和
RequestMappingHandlerAdapter( 處 理 適 配 器 ),可用在Spring-xml.xml配置檔案中使用
mvc:annotation-driven替代註解處理器和介面卡的配置。
同時使用mvc:annotation-driven預設底層就會整合jackson進行物件或集合的json格式字串的轉換。

SpringMVC獲得資料請求的方式

客戶端請求引數的格式是:name=value&name=value… …

伺服器端要獲得請求的引數,有時還需要進行資料的封裝,SpringMVC可以接收如下型別的引數
基本型別引數
POJO型別引數
陣列型別引數
集合型別引數

**

1。獲得基本資料型別的引數

**:
業務方法的引數名稱要與請求引數的名稱一致,引數數值會自動進行對映匹配

案例:
訪問url:http://localhost:8080/spring/quick8?username=zhangsan&age=14

業務方法程式碼

@RequestMapping(value = "/quick8")
    /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save8(String username,int age) throws IOException {
        System.out.println(username);
        System.out.println(age);
    }

服務端輸出
成功獲取到請求引數
在這裡插入圖片描述

**

2.POJO型別引數

**
Controller中的業務方法的POJO引數的屬性名與請求引數的name一致,引數值會自動對映匹配,即自動封裝到一個實體類當中
訪問的url路徑http://localhost:8080/spring/quick8?username=zhangsan&age=14
實體類程式碼

package com.pjh.User;

public class User {
    private int age;
    private String name;

    public User() {
    }

    @Override
    public String toString() {
        return "User{" +
                "age=" + age +
                ", name='" + name + '\'' +
                '}';
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}

Controller類程式碼中的業務方法程式碼

@RequestMapping(value = "/quick9")
    /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save9(User user) throws IOException {
        System.out.println(user);
    }

伺服器端輸出

在這裡插入圖片描述

陣列型別引數
Controller中的業務方法的引數名稱要與請求引數的name一致,引數會自動對映匹配
示例
訪問的urlhttp://localhost:8080/spring/quick10?str=aaa&str=bbb&str=ccc

Controller中的業務方法程式碼

 @RequestMapping(value = "/quick10")
    /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save10(String [] str) throws IOException {
       
        System.out.println(Arrays.asList(str));
    }

服務端輸出

在這裡插入圖片描述

**

3.集合型別引數

**
獲取集合引數時要將集合引數封裝到一個POJO中

以一個提交表單的案例來演示
jsp頁面使用者提交資料

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
</head>
<body>
<form action="${pageContext.request.contextPath}/quick11" method="post">
    <input type="text" name="userList[0].username"><br>
    <input type="text" name="userList[0].age"><br>
    <input type="text" name="userList[1].username"><br>
    <input type="text" name="userList[1].age"><br>
    <input type="submit" value="提交表單">
</form>
</body>
</html>

user類程式碼

package com.pjh.User;

public class User {
    private int age;
    private String username;

    @Override
    public String toString() {
        return "User{" +
                "age=" + age +
                ", username='" + username + '\'' +
                '}';
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }
}

封裝List的vo類程式碼

package com.pjh.User;

import java.util.List;

public class VO {
    private List<User> userList;

    @Override
    public String toString() {
        return "VO{" +
                "userList=" + userList +
                '}';
    }

    public List<User> getUserList() {
        return userList;
    }

    public void setUserList(List<User> userList) {
        this.userList = userList;
    }
}

Controller中的業務方法

@RequestMapping(value = "/quick11")
    /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save11(VO vo) throws IOException {
        System.out.println(vo);
    }

在瀏覽器中提交的資訊

在這裡插入圖片描述

客戶端輸出

在這裡插入圖片描述

**

4.當我們用表單提交中文的資料的時候,會出現亂碼,這時候我們就要配置一個過濾器進行編碼的過濾,同樣是在web.XML中進行配置

**
配置的程式碼

<!--配置過濾器來進行編碼的過濾-->
    <filter>
        <filter-name>CharacterEncodingFilter</filter-name>
        <filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
    <!--配置引數即編碼型別-->
        <init-param>
        <param-name>encoding</param-name>
        <param-value>utf-8</param-value>
    </init-param>
    </filter>
    <filter-mapping>
        <filter-name>CharacterEncodingFilter</filter-name>
        <!--在任何訪問的時候都進行過濾-->
        <url-pattern>/*</url-pattern>
    </filter-mapping>

**

5.當使用ajax提交時,可以指定contentType為json形式,那麼在方法引數位置使用@RequestBody可以直接接受集合資料不需要用POJO進行包裝

**

未過濾前
在這裡插入圖片描述

在這裡插入圖片描述
過濾後
在這裡插入圖片描述

在這裡插入圖片描述

controller類中的業務方法程式碼

@RequestMapping(value = "/quick13")
    /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save13(@RequestBody List<User> userList) throws IOException {
        System.out.println(userList);
    }

訪問業務方法的jsp程式碼

<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
    <title>Title</title>
    <%--這裡使用的是位元組跳動jquery壓縮版引用地址: (速度快推薦!)--%>
    <script src="https://s3.pstatp.com/cdn/expire-1-M/jquery/3.3.1/jquery.min.js"></script>
    <script>
        var array = new  Array();
        array.push({username:"zhangsan",age:"14"});
        array.push({username:"lisi",age:"12"});
        $.ajax({
           type:"POST",
            url:"${pageContext.request.contextPath}/quick13",
            data:JSON.stringify(array),
            contentType:'application/json;charset=utf-8'
        });
    </script>
</head>
<body>
<h1>ajax2</h1>
</body>
</html>

如果你想用其他版本的jquery,可以點選下面的網址尋找https://www.jq22.com/jquery-info122
如果你的的script沒用跳轉到指定的業務方法,很大程度可能是因為jquery的問題,可以使用谷歌的開發者工具看看,狀態顯示200才是成功跳轉了

在這裡插入圖片描述

當有靜態資源需要載入時,比如jquery檔案,如果你SpringMVC的前端控制器DispatcherServlet的url-pattern配置的是/,代表對所有的資源都進行過濾操作,我們可以通過以下兩種方式指定放行靜態資源:

方法二:在spring-mvc.xml配置檔案中指定放行的資源

<mvc:resources mapping="/js/**" location="/js/"/> 

方法1:這是讓tomcat來幫我們找檔案

<mvc:default-servlet-handler/>

**

註解@requestParam的使用

**
註解@RequestParam還有如下引數可以使用:
value:與請求引數名稱一致,配置了這個的 時候方法中的引數名可以隨意配置,不需要與請求引數的名稱一致
required:此在指定的請求引數是否必須包括,預設是true,提交時如果沒有此引數則報錯
defaultValue:當沒有指定請求引數時,則使用指定的預設值賦值
示例

 @ResponseBody
    public void save14(@RequestParam(value = "name" ,required = false,defaultValue = "王五") String username) throws IOException {
        System.out.println(username);
    }

**

6.Restful風格引數

**

Restful是一種架構風格是一種設計風格,而不是一種標準,只是提供了一組設計原則和約束條件。主要用於客戶端和伺服器互動類的軟體,基於這個風格設計的軟體可以更簡潔,更有層次,更易於實現快取機制等

Restful風格的請求是使用“url+請求方式”表示一次請求目的的,HTTP 協議裡面四個表示操作方式的動詞如下:
GET:用於獲取資源
POST:用於新建資源
PUT:用於更新資源
DELETE:用於刪除資源

例如
/user/1 GET : 得到 id = 1 的 user
/user/1 DELETE: 刪除 id = 1 的 user
/user/1 PUT: 更新 id = 1 的 user
/user POST: 新增 user

獲取restful風格的引數

上述url地址/user/1中的1就是要獲得的請求引數,在SpringMVC中可以使用佔位符進行引數繫結。地址/user/1可以寫成/user/{id},佔位符{id}對應的就是1的值。在業務方法中我們可以使用@PathVariable註解進行佔位符的匹配獲取工作

 @RequestMapping(value = "/quick15/{name}")
    /* 告訴SpringMvc框架,方法返回的字串不是跳轉是直接在http響應體中返回*/
    @ResponseBody
    public void save15(@PathVariable(value = "name" ,required = false) String username) throws IOException {
        System.out.println(username);
    }

上述程式碼中的{name}就是佔位符,@PathVariable註解中的value值要和佔位符裡的名稱一樣

**

7.自定義型別轉換器

**

SpringMVC 預設已經提供了一些常用的型別轉換器,例如客戶端提交的字串轉換成int型進行引數設定。
但是不是所有的資料型別都提供了轉換器,沒有提供的就需要自定義轉換器,例如:日期型別的資料就需要自定義轉換器。

自定義型別轉換器的開發步驟:
1.定義轉換器類實現Converter介面
2.在配置檔案中宣告轉換器
3.在中引用轉換器

1.定義轉換器類實現Converter介面

package com.pjh.Converter;
import org.springframework.core.convert.converter.Converter;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Date;
public class DataConvert implements Converter<String, Date> {
    public Date convert(String source) {
        SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd");
        Date parse = null;
        try {
            parse = simpleDateFormat.parse(source);
        } catch (ParseException e) {
            e.printStackTrace();
        }
        return parse;
    }
}

2.在配置檔案中宣告轉換器

<bean id="ConversionServiceConverter" class="org.springframework.context.support.ConversionServiceFactoryBean">
    <property name="converters">
        <list>
            <bean class="com.pjh.Converter.DataConvert"/>
        </list>
    </property>
    </bean>

3.在中引用轉換器

 <mvc:annotation-driven conversion-service="ConversionServiceConverter"/>

**

8.獲取請求頭

**

使用@CookieValue可以獲得指定Cookie的值
@CookieValue註解的屬性如下:
value:指定cookie的名稱
required:是否必須攜帶此cookie

獲取Cookie,有專門註解

RequestMapping("/quick18")
@ResponseBody
public void quickMethod18( @CookieValue(value = "JSESSIONID",required = false) String jsessionid)
{   
System.out.println(jsessionid);
}

獲取User-Agent沒有

@RequestMapping("/quick17")
@ResponseBody
public void quickMethod17(      @RequestHeader(value = "User-Agent",required = false) String headerValue)
{  
System.out.println(headerValue);
}

以上就是SpringMVC系列的第二章,主要側重SpringMVC的請求與資料響應的操作,我後續會持續更新Springmvc系列課程,本部落格主要側重於資料結構於演算法和java開發,覺得我的文章有幫助的小夥伴可以關注我,有疑問可評論私信,相逢即是緣,大家高處見

在這裡插入圖片描述

相關文章