SpringMVC超詳細(xì)講解視圖和視圖解析器
SpringMVC-視圖和視圖解析器
1.基本介紹
在 springMVC 中的目標(biāo)方法最終返回都是一個(gè)視圖(有各種視圖).
返回的視圖都會(huì)由一個(gè)視圖解析器來處理 (視圖解析器有很多種)
2.自定義視圖
1.為什么要自定義視圖
在默認(rèn)情況下,我們都是返回默認(rèn)的視圖, 然后這個(gè)返回的視圖交由 SpringMVC 的 InternalResourceViewResolver 視圖處理器來處理的
<bean id="internalResourceViewResolver" class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/pages/"/>
<property name="suffix" value=".jsp"/>
</bean>
在實(shí)際開發(fā)中,我們有時(shí)需要自定義視圖,這樣可以滿足更多更復(fù)雜的需求
2.自定義視圖實(shí)例-代碼實(shí)現(xiàn)
1.配置 springDispatcherServlet-servlet.xml , 增加自定義視圖解析器
<!--
1. 配置自定義視圖解析器BeanNameViewResolver
2. BeanNameViewResolver可以去解析我們自定義的視圖
3. 配置 屬性 order, 表示視圖解析器執(zhí)行的順序, 值越小, 優(yōu)先級(jí)越高
4. 屬性 order 的默認(rèn)值是最低優(yōu)先級(jí) ,值為 Integer.MAX_VALUE
int LOWEST_PRECEDENCE = 2147483647
-->
<bean class="org.springframework.web.servlet.view.BeanNameViewResolver">
<property name="order" value="99"/>
</bean>
2.創(chuàng)建自定義視圖類
@Component(value = "view")
public class MyView extends AbstractView {
@Override
protected void renderMergedOutputModel(Map<String, Object> model,
HttpServletRequest request,
HttpServletResponse response) throws Exception {
//完成視圖渲染
//并且可以確定我們要跳轉(zhuǎn)的頁面 [請(qǐng)求轉(zhuǎn)發(fā)] /WEB-INF/pages/my_view.jsp
System.out.println("進(jìn)入到自己的視圖..");
Object o = model.get("name");
//llp
System.out.println(o);
//1. 下面就是進(jìn)行請(qǐng)求轉(zhuǎn)發(fā)到 /WEB-INF/pages/my_view.jsp
//2. /WEB-INF/pages/my_view.jsp 會(huì)被springmvc解析
// /springmvc/WEB-INF/pages/my_view.jsp
request.getRequestDispatcher("/WEB-INF/pages/my_view.jsp").forward(request, response);
}
}3.創(chuàng)建goodsHandler測試
@Controller
@RequestMapping("goods")
public class GoodsHandler {
@RequestMapping(value = "/buy")
public String byGoods(Map<String,Object> map){
map.put("name","llp");
return "view";
}
}4.創(chuàng)建my_view.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>my_view頁面</title>
</head>
<h1>進(jìn)入到my_view頁面</h1>
<p>是從自定義視圖來的..</p>
<body>
</body>
</html>5.測試效果

3.自定義視圖工作流程小結(jié)
自定義視圖-小結(jié)
自定義視圖: 創(chuàng)建一個(gè) View 的 bean, 該 bean 需要繼承自 AbstractView, 并實(shí)現(xiàn) renderMergedOutputModel 方法.
并把自定義 View 加入到 IOC 容器中
自定義視圖的視圖處理器,使用 BeanNameViewResolver, 這個(gè)視圖處理器也需要配置 到 ioc 容器
BeanNameViewResolver 的調(diào)用優(yōu)先級(jí)需要設(shè)置一下,設(shè)置 order 比 Integer.MAX_VAL 小的值. 以確保其在 InternalResourceViewResolver 之前被調(diào)用
自定義視圖-工作流程
SpringMVC 調(diào)用目標(biāo)方法, 返回自定義 View 在 IOC 容器中的 id
SpringMVC 調(diào)用 BeanNameViewResolver 解析視圖: 從 IOC 容器中獲取 返回 id 值對(duì)應(yīng)的 bean, 即自定義的 View 的對(duì)象
SpringMVC 調(diào)用自定義視圖的 renderMergedOutputModel 方法渲染視圖
說明: 如果在 SpringMVC 調(diào)用目標(biāo)方法, 返回自定義 View 在 IOC 容器中的 id, 不存在, 則仍然按照默認(rèn)的視圖處理器機(jī)制處理
如果將默認(rèn)視圖解析器設(shè)置的優(yōu)先級(jí)比自定義視圖高,默認(rèn)視圖解析器不管頁面是否存在都會(huì)直接返回,不會(huì)在走自定義視圖解析器。
自定義 View在IOC容器中的 id存在時(shí),自定義視圖解析器執(zhí)行流程:


自定義 View在IOC容器中的 id不存在時(shí),自定義視圖解析器執(zhí)行流程
通用會(huì)先執(zhí)行BeanNameViewResolver,可以看到此時(shí),容器中不包含id=”view”的bean

回到DispatcherServlet我們可以看到 List viewResolvers列表包含兩個(gè)視圖解析器,一個(gè)時(shí)自定義的視圖還有一個(gè)是默認(rèn)的視圖解析器。如果返回的自定義視圖為空,會(huì)繼續(xù)遍歷最終初始化默認(rèn)視圖解析器從而按照默認(rèn)視圖解析器的處理機(jī)制繼續(xù)執(zhí)行,不管/WEB-INF/pages/view.jsp 是否存在程序都會(huì)返回


4.目標(biāo)方法直接指定轉(zhuǎn)發(fā)或重定向
1.使用實(shí)例
目標(biāo)方法中指定轉(zhuǎn)發(fā)或者重定向
默認(rèn)返回的方式是請(qǐng)求轉(zhuǎn)發(fā),然后用視圖處理器進(jìn)行處理,比如在目標(biāo)方法中這樣寫:
@Controller
public class LoginServlet {
@RequestMapping(value = "/login")
public String login(){
System.out.println("login....");
return "login_ok";
}
}也可以在目標(biāo)方法直接指定重定向或轉(zhuǎn)發(fā)的 url 地址
如果指定重定向,不能定向到 /WEB-INF 目錄中
應(yīng)用實(shí)例-代碼實(shí)現(xiàn)
修改 GoodsHandler.java, 增加方法 order()
/**
* 演示直接指定要請(qǐng)求轉(zhuǎn)發(fā)的或者是重定向的頁面
* @return
*/
@RequestMapping(value = "/order")
public String order() {
System.out.println("=======order()=====");
//請(qǐng)求轉(zhuǎn)發(fā)到 /WEB-INF/pages/my_view.jsp
//下面的 /WEB-INF/pages/my_view.jsp 被解析成 /springmvc/WEB-INF/pages/my_view.jsp
//return "forward:/WEB-INF/pages/my_view.jsp";
//return "forward:/aaa/bbb/ok.jsp";
//直接指定要重定向的頁面
//1. 對(duì)于重定向來說,不能重定向到 /WEB-INF/ 目錄下
//2. redirect 關(guān)鍵字,表示進(jìn)行重定向
//3. /login.jsp 在服務(wù)器解析 /springmvc/login.jsp
return "redirect:/login.jsp";
// /WEB-INF/pages/my_view.jsp 被解析 /springmvc/WEB-INF/pages/my_view.jsp
//return "redirect:/WEB-INF/pages/my_view.jsp";
}2.修改view.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>自定義視圖測試</title>
</head>
<body>
<h1>自定義視圖測試</h1>
<a href="goods/buy" rel="external nofollow" >點(diǎn)擊到自定義視圖-</a><br/>
<a href="goods/order" rel="external nofollow" >測試在目標(biāo)方法中指定請(qǐng)求轉(zhuǎn)發(fā)或者重定向的頁面-</a><br/>
</body>
</html>
2.指定請(qǐng)求轉(zhuǎn)發(fā)流程-Debug源碼
1.SpirngMVC默認(rèn)情況下
/**
* 演示直接指定要請(qǐng)求轉(zhuǎn)發(fā)的或者是重定向的頁面
* @return
*/
@RequestMapping(value = "/order")
public String order() {
System.out.println("=======order()=====");
return "my_view.jsp";
}
可以看到默認(rèn)轉(zhuǎn)發(fā)是走的InternalResourceView默認(rèn)視圖解析器,beanName=my_view.jsp 對(duì)應(yīng)的實(shí)例則是InternalResourceView

最終執(zhí)行到InternalResourceView的renderMergedOutputModel方法中dispatcherPath=/WEB-INF/pages/my_view.jsp.jsp進(jìn)行請(qǐng)求轉(zhuǎn)發(fā)

2.指定forward關(guān)鍵字
/**
* 演示直接指定要請(qǐng)求轉(zhuǎn)發(fā)的或者是重定向的頁面
* @return
*/
@RequestMapping(value = "/order")
public String order() {
System.out.println("=======order()=====");
return "forward:/WEB-INF/pages/my_view.jsp";
}
在DispatcherServlet中我們可以看到,當(dāng)我們指定了forward關(guān)鍵字時(shí),viewResolver還是默認(rèn)的InternalResourceView,和不指定forward關(guān)鍵字不同的地方子在于,指定了關(guān)鍵字后beanName變成了forward: 而url則是我們后面指定的路徑,因此我們?cè)谑褂胒orward關(guān)鍵字進(jìn)行請(qǐng)求轉(zhuǎn)發(fā)時(shí),SpringMVC底層會(huì)根據(jù)程序員在后面指定的路徑進(jìn)行請(qǐng)求轉(zhuǎn)發(fā),如果設(shè)置的路徑不存在則會(huì)拋出404NOTFOUND異常


3.指定重定向流程-Debug源碼
/**
* 演示直接指定要請(qǐng)求轉(zhuǎn)發(fā)的或者是重定向的頁面
* @return
*/
@RequestMapping(value = "/order")
public String order() {
System.out.println("=======order()=====");
return "redirect:/login.jsp";
}
在前面我們配置了自定義視圖和默認(rèn)視圖解析器,可以看到這里走的是默認(rèn)的視圖解析器。redirect: 被作為beanName, RedirectView則是對(duì)應(yīng)的bean實(shí)例

從里可以看出在SpringMvc中,重定向"redirect:/login.jsp" ,斜杠被瀏覽器解析成ip:port,而在服務(wù)端springmvc底層RedirectView會(huì)解析成“/springmvc/login.jsp”

RedirectView
@Override
protected void renderMergedOutputModel(Map<String, Object> model, HttpServletRequest request,
HttpServletResponse response) throws IOException {
String targetUrl = createTargetUrl(model, request);
targetUrl = updateTargetUrl(targetUrl, model, request, response);
// Save flash attributes
RequestContextUtils.saveOutputFlashMap(targetUrl, request, response);
// Redirect 重定向
sendRedirect(request, response, targetUrl, this.http10Compatible);
}
到此這篇關(guān)于SpringMVC超詳細(xì)講解視圖和視圖解析器的文章就介紹到這了,更多相關(guān)SpringMVC視圖內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
MyBatis將查詢出的兩列數(shù)據(jù)裝配成鍵值對(duì)的操作方法
這篇文章主要介紹了MyBatis將查詢出的兩列數(shù)據(jù)裝配成鍵值對(duì)的操作代碼,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-08-08
springcloud?如何解決微服務(wù)之間token傳遞問題
這篇文章主要介紹了springcloud?如何解決微服務(wù)之間token傳遞問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
springboot項(xiàng)目配置多個(gè)kafka的示例代碼
這篇文章主要介紹了springboot項(xiàng)目配置多個(gè)kafka,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-04-04
SpringBoot請(qǐng)求參數(shù)傳遞與接收說明小結(jié)
這篇文章主要介紹了SpringBoot請(qǐng)求參數(shù)傳遞與接收,本文通過示例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-12-12
Spring中的事務(wù)管理及實(shí)現(xiàn)方式解析
這篇文章主要介紹了Spring中的事務(wù)管理及實(shí)現(xiàn)方式解析,Spring事務(wù)管理基于底層數(shù)據(jù)庫本身的事務(wù)處理機(jī)制,數(shù)據(jù)庫事務(wù)的基礎(chǔ),是掌握Spring事務(wù)管理的基礎(chǔ),這篇總結(jié)下Spring事務(wù),需要的朋友可以參考下2024-01-01
非maven項(xiàng)目快速轉(zhuǎn)換為maven項(xiàng)目的方法步驟
時(shí)候我們導(dǎo)入的項(xiàng)目并不是有maven來管理依賴的,而是要手動(dòng)添加jar包,比較麻煩,本文主要介紹了非maven項(xiàng)目快速轉(zhuǎn)換為maven項(xiàng)目的方法步驟,具有一定的參考價(jià)值,感興趣的可以了解一下2024-01-01

