SpringMVC的響應處理


本文環境搭建參考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標籤配置不過濾的資源:

  1. location元素表示webapp目錄下的包下的所有文件
  2. 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>
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章