简单跳转专题
个人建议重新练习一遍搭建的过程,如果感觉麻烦你可以直接复制上一个工程,但是需要修改pom.xml中的一点信息
1
2
3
|
< groupId >com.hanpang.springmvc</ groupId > < artifactId >springmvc-demo01</ artifactId > < version >0.0.1-SNAPSHOT</ version > |
1.核心配置类和加载类
1
2
3
4
5
6
7
8
9
10
|
package com.hanpang.config; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.config.annotation.EnableWebMvc; @Configuration @EnableWebMvc @ComponentScan (basePackages= "com.hanpang.**.web" ) public class WebConfig { } |
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
package com.hanpang.config; import org.springframework.web.servlet.support.AbstractAnnotationConfigDispatcherServletInitializer; public class WebInitializer extends AbstractAnnotationConfigDispatcherServletInitializer { @Override protected Class<?>[] getRootConfigClasses() { return new Class[] {WebConfig. class }; } @Override protected Class<?>[] getServletConfigClasses() { return null ; } @Override protected String[] getServletMappings() { return new String[] { "/" }; } } |
2.JavaWeb阶段的跳转方式
请注意SpringMVC的方法中的形参,框架给我们完成了实例化的操作
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
|
package com.hanpang.web; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; @Controller //告知其是一个控制器 public class Demo01Controller { @RequestMapping (path= "/test01" ) public ModelAndView 传统方式跳转_请求转发(HttpServletRequest request,HttpServletResponse response) throws ServletException, IOException { System.out.println( "默认对形参的进行了实例化操作" ); request.getRequestDispatcher( "/WEB-INF/jsp/demo01.jsp" ).forward(request, response); return null ; } @RequestMapping (path= "/test02" ) public ModelAndView 传统方式跳转_重定向(HttpServletRequest request,HttpServletResponse response) throws IOException { System.out.println( "默认对形参的进行了实例化操作" ); response.sendRedirect(request.getContextPath()+ "/view/result01.jsp" ); return null ; } } |
NOTE: 这种方式我们几乎不再使用了,只是只是简单的演示和回顾一下,至少我们可以使用这种方式获取Servlet API!!!
3.Controller跳转到JSP的方式演示
在示例最后的时候,我们会加入JSP的视图解析器,开始阶段我们还是按照传统的方式,有一个循序渐进的过程
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
|
package com.hanpang.web; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; @Controller //告知其是一个控制器 public class Demo01Controller { @RequestMapping (path= "/test03" ) public ModelAndView 默认情况下是请求转发(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "/WEB-INF/jsp/demo01.jsp" ); return mav; } @RequestMapping (path= "/test04" ) public ModelAndView 设置重定向的方式(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "redirect:/view/result01.jsp" ); //或者 //mav.setViewName(UrlBasedViewResolver.REDIRECT_URL_PREFIX+"/view/result01.jsp"); return mav; } } |
感觉跟Java Web阶段的方式差不多,只是重定向的时候设置了一个简单的前缀
4.Controller跳转到Controller的方式演示
类似于从一个Servlet调到另一个Servlet
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
|
package com.hanpang.web; import java.io.IOException; import javax.servlet.ServletException; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.servlet.ModelAndView; @Controller //告知其是一个控制器 public class Demo01Controller { @RequestMapping (path= "/test05" ) public ModelAndView 直接设置映射路径即可(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "/test03" ); return mav; } @RequestMapping (path= "/test06" ) public ModelAndView 设置重定向(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "redirect:/test04" ); return mav; } } |
5.加入JSP的视图解析器
上面的演示过程中,我们发现ModelAndView中setViewName是用来完成跳转的,这里面传递的数据是字符串,但是处理方式不太一样,当跳转的路径有前缀redirect:的时候,那么处理方式不一样.
还有,如果我们有多个类似/WEB-INF/jsp/demo01.jsp形式的字符串的时候,我们发现公共的部分很多,SpringMVC给我们提供了一个专门处理 Controller请求转发JSP页面 的类.
请注意我的描述: 如果发现你传递的字符串没有设置任何前缀标识,那么默认情况下使用配置JSP视图解析器处理,并且完成请求转发的操作
注解:配置核心配置类
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
package com.hanpang.config; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.ViewResolver; import org.springframework.web.servlet.config.annotation.EnableWebMvc; import org.springframework.web.servlet.view.InternalResourceViewResolver; import org.springframework.web.servlet.view.JstlView; @Configuration @EnableWebMvc @ComponentScan (basePackages= "com.hanpang.**.web" ) public class WebConfig { @Bean //实例化 public ViewResolver viewResolver() { InternalResourceViewResolver jspViewResolver = new InternalResourceViewResolver(); jspViewResolver.setViewClass(JstlView. class ); //springmvc支持jstl标签 jspViewResolver.setPrefix( "/WEB-INF/" ); jspViewResolver.setSuffix( ".jsp" ); return jspViewResolver; } } |
**NOTE:**请注意方法上的注解@Bean
方法等价于XML中的代码如下
1
2
3
4
5
|
< bean id = "jspResourceViewResolver" class = "org.springframework.web.servlet.view.InternalResourceViewResolver" > < property name = "prefix" value = "/WEB-INF/" /> < property name = "suffix" value = ".jsp" /> < property name = "viewClass" value = "org.springframework.web.servlet.view.JstlView" /> </ bean > |
改进Controller跳转JSP代码
该视图解析器只能针对于JSP的请求转发,对重定向无效,请注意代码的注释内容
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
@RequestMapping (path= "/test03" ) public ModelAndView 默认情况下是请求转发(){ ModelAndView mav = new ModelAndView(); //mav.setViewName("/WEB-INF/jsp/demo01.jsp"); //改进后:默认情况下,会使用JSP视图解析器处理, // prefix+"jsp/demo01"+suffix => /WEB-INF/jsp/demo01.jsp mav.setViewName( "jsp/demo01" ); //发现字符串没有前缀修饰 return mav; } @RequestMapping (path= "/test04" ) public ModelAndView 设置重定向的方式(){ ModelAndView mav = new ModelAndView(); //解析器对重定向无效 mav.setViewName( "redirect:/view/result01.jsp" ); return mav; } |
改进Controller跳转Controller代码
1
2
3
4
5
6
|
@RequestMapping (path= "/test05" ) public ModelAndView 直接设置映射路径即可(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "/test03" ); return mav; } |
当配置完JSP视图解析器后,对上述的代码再次进行测试,查看访问结果有惊喜
符合我们之前说的"/test03"是一个字符串,默认情况下会使用JSP视图解析器进行处理,那么如何改进呢?
可以设置前缀"forward:",改进代码如下:
1
2
3
4
5
6
7
8
|
@RequestMapping (path= "/test05" ) public ModelAndView 直接设置映射路径即可(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "forward:/test03" ); //或者 //mav.setViewName(UrlBasedViewResolver.FORWARD_URL_PREFIX+"/test03"); return mav; } |
当发现字符串使用forward:修饰后,处理情况改变为从Controller请求转换到另一个Controller,而如果做到重定向的话,代码如下:
1
2
3
4
5
6
|
@RequestMapping (path= "/test06" ) public ModelAndView 设置重定向(){ ModelAndView mav = new ModelAndView(); mav.setViewName( "redirect:/test04" ); return mav; } |
6.InternalResourceViewResolver 附录
InternalResourceViewResolver:它是URLBasedViewResolver的子类,所以URLBasedViewResolver支持的特性它都支持。
在实际应用中InternalResourceViewResolver也是使用的最广泛的一个视图解析器,那么InternalResourceViewResolver有什么自己独有的特性呢?
单从字面意思来看,我们可以把InternalResourceViewResolver解释为内部资源视图解析器,这就是InternalResourceViewResolver的一个特性。
InternalResourceViewResolver会把返回的视图名称都解析为InternalResourceView对象,InternalResourceView会把Controller处理器方法返回的模型属性都存放到对应的request属性中,然后通过RequestDispatcher在服务器端把请求forword重定向到目标URL。
比如在InternalResourceViewResolver中定义了prefix=/WEB-INF/,suffix=.jsp,然后请求的Controller处理器方法返回的视图名称为test,那么这个时候InternalResourceViewResolver就会把test解析为一个InternalResourceView对象,先把返回的模型属性都存放到对应的HttpServletRequest属性中,然后利用RequestDispatcher在服务器端把请求forword到/WEB-INF/test.jsp。这就是InternalResourceViewResolver一个非常重要的特性,我们都知道存放在/WEB-INF/下面的内容是不能直接通过request请求的方式请求到的,为了安全性考虑,我们通常会把jsp文件放在WEB-INF目录下,而InternalResourceView在服务器端跳转的方式可以很好的解决这个问题。下面是一个InternalResourceViewResolver的定义,根据该定义当返回的逻辑视图名称是test的时候,InternalResourceViewResolver会给它加上定义好的前缀和后缀,组成“/WEB-INF/test.jsp”的形式,然后把它当做一个InternalResourceView的url新建一个InternalResourceView对象返回。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持服务器之家。
原文链接:https://juejin.im/post/5aa8706ff265da239b411d59