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一致,引數會自動對映匹配
示例
訪問的url:http://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
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);
}