目錄
本文環境搭建參考SpringMVC入門。
-> 本文源碼
一、返回值分類
1.返回字符串
Controller方法返回字符串,可以指定邏輯視圖的名稱,根據視圖解析器轉爲物理視圖的地址。
如指定邏輯視圖名"success",可經過視圖解析器解析爲 jsp 物理路徑: /WEB-INF/pages/success.jsp
最簡示例:
@RequestMapping(value="/hello")
public String sayHello() {
System.out.println("Hello SpringMVC!!");
// 跳轉到XX頁面
return "success";
}
根據方法返回的"success",可將頁面跳轉至如下jsp:
應用示例:
頁面通過post請求,更新數據庫中的數據,返回"update",並跳轉至相應頁面。
@Controller
@RequestMapping("/user")
public class UserController {
/**
* 請求參數的綁定
*/
@RequestMapping(value="/initUpdate")
public String initUpdate(Model model) {
// 模擬從數據庫中查詢的數據
User user = new User();
user.setUsername("張三");
user.setPassword("123");
user.setMoney(100d);
user.setBirthday(new Date());
model.addAttribute("user", user);
return "update";
}
}
<h3>修改用戶</h3>
${ requestScope }
<form action="user/update" method="post">
姓名:<input type="text" name="username" value="${ user.username }"><br>
密碼:<input type="text" name="password" value="${ user.password }"><br>
金額:<input type="text" name="money" value="${ user.money }"><br>
<input type="submit" value="提交">
</form>
2. 返回void
如果控制器的方法返回值編寫成void,執行程序報404的異常,說明默認查找JSP頁面沒有找到。
因爲默認會跳轉到@RequestMapping(value="/initUpdate") 所指定的 user/initUpdate.jsp這個頁面,如果我們沒配這個資源,自然會報404了。
爲了解決這種尷尬的情況,我們可以使用請求轉發或者重定向跳轉到指定的頁面:
@RequestMapping(value="/initAdd")
public void initAdd(HttpServletRequest request,HttpServletResponse response) throws
Exception {
System.out.println("請求轉發或者重定向");
// 請求轉發
// request.getRequestDispatcher("/WEB-INF/pages/add.jsp").forward(request,
response);
// 重定向
// response.sendRedirect(request.getContextPath()+"/add2.jsp");
response.setCharacterEncoding("UTF-8");
response.setContentType("text/html;charset=UTF-8");
// 直接響應數據
response.getWriter().print("你好");
return;
}
3. 返回ModelAndView對象
ModelAndView 是 SpringMVC 爲我們提供的一個對象,該對象也可以用作控制器方法的返回值。
該對象中有兩個方法:
addObject()方法可以添加模型到該對象中,並可在頁面上用EL表達式獲取:${attributeName}。
setViewName()方法用於設置邏輯視圖名稱,視圖解析器會根據名稱前往指定的視圖。
示例代碼:
/**
* 返回ModelAndView
* @return
*/
@RequestMapping("/testModelAndView")
public ModelAndView testModelAndView(){
// 創建ModelAndView對象
ModelAndView mv = new ModelAndView();
System.out.println("testModelAndView方法執行了...");
// 模擬從數據庫中查詢出User對象
User user = new User();
user.setUsername("小豬");
user.setPassword("111456");
user.setAge(25);
// 把user對象存儲到mv對象中,同時也會把user對象存入到request對象
mv.addObject("user",user);
// 跳轉到哪個頁面
mv.setViewName("success");
return mv;
}
結果頁面:
<body>
<h3>執行成功</h3>
${user.username}
${user.password}
</body>
二、SpringMVC提供的轉發和重定向
SpringMVC提供了使用關鍵字來進行請求轉發和重定向的能力。
- forward請求轉發
在controller方法返回String類型的情況下,想進行請求轉發也可以編寫成如下形式:
/**
* 使用forward關鍵字進行請求轉發
* "forward:轉發的JSP路徑",不走視圖解析器了,所以需要編寫完整的路徑
* @return
* @throws Exception
*/
@RequestMapping("/delete")
public String delete() throws Exception {
System.out.println("delete方法執行了...");
// return "forward:/WEB-INF/pages/success.jsp";
return "forward:/user/findAll";
}
需要注意的是,如果用了 forward: 則路徑必須寫成實際視圖 url,不能寫邏輯視圖。
它相當於“ request.getRequestDispatcher(“url”).forward(request,response)” 。
使用請求轉發,既可以轉發到 jsp,也可以轉發到其他的控制器方法。
- redirect 重定向
/**
* 重定向
* @return
* @throws Exception
*/
@RequestMapping("/count")
public String count() throws Exception {
System.out.println("count方法執行了...");
return "redirect:/add.jsp";
// return "redirect:/user/findAll";
}
請求轉發與重定向都不需要寫完整路徑,因爲SpringMVC底層已經進行了處理。
三、通過ResponseBody 響應 json 數據
@ResponseBody註解 用於將 Controller 的方法返回的對象,通過 HttpMessageConverter 接口轉換爲指定格式的數據如: json,xml 等,通過 Response 響應給客戶端。
1.配置不攔截靜態資源
DispatcherServlet會攔截到所有的資源,導致靜態資源(img、css、js)也會被攔截,從而不能被使用。
解決方法是需要配置靜態資源不進行攔截,在springmvc.xml配置文件添加如下配置:
用mvc:resources標籤配置不過濾的資源:
- location元素表示webapp目錄下的包下的所有文件
- mapping元素表示以/static開頭的所有請求路徑,如/static/a 或者/static/a/b
<!-- 設置靜態資源不過濾 -->
<mvc:resources location="/css/" mapping="/css/**"/> <!-- 樣式 -->
<mvc:resources location="/images/" mapping="/images/**"/> <!-- 圖片 -->
<mvc:resources location="/js/" mapping="/js/**"/> <!-- javascript -->
2.使用@RequestBody獲取請求體數據
下面是jsp中的代碼,用於發送ajax請求:
<head>
<title>Title</title>
<script src="js/jquery.min.js"></script>
<script>
// 頁面加載,綁定單擊事件
$(function(){
$("#btn").click(function(){
// alert("hello btn");
// 發送ajax請求
$.ajax({
// 編寫json格式,設置屬性和值
url:"user/testAjax",
contentType:"application/json;charset=UTF-8",
data:'{"username":"hehe","password":"123","age":30}',
dataType:"json",
type:"post",
success:function(data){
// data服務器端響應的json的數據,進行解析
alert(data);
alert(data.username);
alert(data.password);
alert(data.age);
}
});
});
});
</script>
</head>
<body>
<button id="btn">發送ajax的請求</button>
</body>
控制器方法:
在入參前使用@RequestBody註解來獲取請求體中的數據——
/**
* 獲取請求體的數據
* @param body
*/
@RequestMapping("/testAjax")
public void testAjax(@RequestBody String body) {
System.out.println(body);
}
3.使用@RequestBody把json轉換成JavaBean
@RequestMapping("/testAjax")
public void User testAjax(@RequestBody User user){
System.out.println("testAjax方法執行了...");
// 客戶端發送ajax的請求,傳的是json字符串,後端把json字符串封裝到user對象中
System.out.println(user);
}
4.使用@ResponseBody把JavaBean轉換成json直接響應
/**
* 模擬異步請求響應
*/
@RequestMapping("/testAjax")
public @ResponseBody User testAjax(@RequestBody User user){
System.out.println("testAjax方法執行了...");
// 客戶端發送ajax的請求,傳的是json字符串,後端把json字符串封裝到user對象中
System.out.println(user);
// 做響應,模擬查詢數據庫
user.setUsername("haha");
user.setAge(40);
// 做響應
return user;
}
注意:json和JavaBean互相轉換的過程中,需要使用jackson的jar包
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-databind</artifactId>
<version>2.9.0</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-core</artifactId>
<version>2.9.0</version>
</dependency>
<dependency>
<groupId>com.fasterxml.jackson.core</groupId>
<artifactId>jackson-annotations</artifactId>
<version>2.9.0</version>
</dependency>