最近再使用SpringMVC进行页面跳转的时候,不知道发生了什么,始终都无法正确跳转。后来问题解决了,发现是对于转发和重定向没有能很好的理解,以此写篇博客,权当做积累了!
声明:本博客的所有代码,均为模拟问题的代码,将问题从复杂的系统中抽取出来,进行分析!
一、基本配置
备注:只截取了几个比较关键的配置点
SpringMVC配置:
<span style="font-family:KaiTi_GB2312;font-size:18px;"><bean
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/jsp/"></property>
<property name="suffix" value=".jsp" />
</bean></span>
web.xml配置
<span style="font-family:KaiTi_GB2312;font-size:18px;"> <display-name>Angel-web</display-name>
<welcome-file-list>
<welcome-file>/ccweb/jsp/haha/Ynna.jsp</welcome-file>
</welcome-file-list></span><pre name="code" class="html"><span style="font-family:KaiTi_GB2312;font-size:18px;"> <!-- springmvc的前端控制器 -->
<servlet>
<servlet-name>Angel-web</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:spring/springmvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>Angel-web</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping></span>
系统结构图:需要注意的是,我们平时大多是将JSP文件夹放入WEB-INF文件夹下,然后在SpringMVC的前缀配置中,是配置<property name="prefix" value="/WEB-INF/jsp/"></property>,但是这里不同!
备注:并不建议将jsp页面放入其他位置,但这个根据情况而定,WEB-INF文件并不被默认认为是web文件,用户无法直接访问到这里面的文件源码,增加了一定的安全性!如果将jsp文件或者其他文件植入系统的其他目录,那么用户可能直接越过Struts或者SpringMVC的controller进行访问。
有兴趣的可以查一下,更多了解为什么最好是将文件放入到WEB-INF文件夹下!
二、问题分析
出现的问题是:在controller中,通过常用的如下方式:
<span style="font-family:KaiTi_GB2312;font-size:18px;">@RequestMapping("/test")
public String findYnna() {
return "Ynna";
}</span>
可以访问到这个controller方法,但无法跳转到Ynna这个jsp页面,不仅如此,连直接访问具体路径的:return "ccweb/jsp/Ynna.jsp"; 这种方式,也无法打开这个jsp页面。但神奇的是,如果输入http://localhost:8080/ccweb......这种形式,却又可以正常打开这个页面(这也就是为什么建议将文件植入到WEB-INF下面的一个原因,一般来说,用户只能通过访问MVC中的Controller,根据其处理结果调用对应的视图解析器进行视图显示)。
出现这种情况,第一个想到的是,SpringMVC的前缀后缀配置问题,但是如文章第一部分,前缀和后缀都配得很好。问题:(问了一下项目中的实际应用,大都采用return "ccweb/jsp/Ynna.jsp"方式进行页面跳转,我就是想知道,将JSP页面移出了WEB-INF文件夹,通过文件的相对路径直接进行访问页面,既然如此,那配个前缀和后缀是干嘛使的????????谁能告诉我?????????)
然后,检查了SpringMVC配置,发现没问题,第二个想到的是web.xml里面对于Servlet请求拦截的配置:
<span style="font-family:KaiTi_GB2312;font-size:18px;"><servlet-mapping>
<servlet-name>Angel-web</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping></span>
最终发现,拦截配置的也没有问题,从系统名称后的 / 开始拦截,有时候,会加以特殊的写法,比如在Struts里面,个人就偏向于,拦截 /*.do形式的。可是现在的问题是:页面访问不到,到底是出现了什么问题?????
检查完了这一步,紧接着是检查如果直接在jsp页面,通过window.location.href方法或者说添加<a href="***">标签的形式,通过植入相应jsp页面的具体相对路径,也就是ccweb/jsp/Ynna.jsp这种形式,是否可以正常打开? 结果是:可以!
那么问题就来了,以上验证说明,这个页面是完全可以通过相对路径进行直接访问和跳转的,那么,问题再次被定位到controller在匹配视图解析这个过程!
后来的解决方法是:
<span style="font-family:KaiTi_GB2312;font-size:18px;">return "redirect:ccweb/jsp/Ynna.jsp"; </span>
可以明显的看到这里用了redirect关键字:重定向!(这里只是实现页面的跳转,也可以用forward,根据具体的需求而定,有不了解的,可以参考博客:
[drp 7]转发和重定向的区别
)
再次分析这个问题:当前页面pageOne——通过触发一个事件——访问controller方法——跳转到另一个页面(重定向)或者在当前页面打开另一个页面(转发),而之前没有使用redirect的时候,是纯粹的打开一个页面!说到这里可能就有点懵,但是,请分析以下现象:
1:做过后台管理系统的人,应该都不陌生,左边一个菜单栏,然后点击一个,在布局的右边位置打开一个页面(可以类比iframe布局或者html5中的header、nav、footer标签什么的)——打开一个页面
2:同样是在布局的右边位置,通过触发一个事件,比如说button的单击事件,再通过后台的处理,最终在右边位置打开另一个页面——打开一个页面
3:1和2同样是打开一个页面,但是,这是不是相同的概念?????
三、总结
虽说问题比较小,但是也还是要总结的。这次的问题暴露,让自己对于SpringMVC的工作原理,以及重定向和转发都有了再一次不同的理解。另外,感激抛给我这个问题的小伙伴,大家一起成长!不过,话说回来,所谓的在项目中成长,其实不是让你去像个machine一样不停的做项目,或者说,你老人家真的该总结发现和思考了!对于原理上的东西不熟悉,真的可以导致你整个人都是蒙的!做machine还是human ,自己选吧!
PS:尤其是那个谁,我都写了篇博客总结,你老人家呢?