需求:前端浏览器请求 url : http://localhost:8080/demo/handle01 ,前端⻚⾯显示后台服务器的时间
开发过程
1 )配置 DispatcherServlet 前端控制器
2 )开发处理具体业务逻辑的 Handler ( @Controller 、 @RequestMapping )
3 ) xml 配置⽂件配置 controller 扫描,配置 springmvc 三⼤件
4 )将 xml ⽂件路径告诉 springmvc ( DispatcherServlet )
创建web-app项目
在pom.xml中引入spring-webmvc依赖, tomcat插件,
<dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>5.3.5</version></dependency>
<build>
<plugins>
<plugin>
<groupId>org.apache.tomcat.maven</groupId>
<artifactId>tomcat7-maven-plugin</artifactId>
<version>2.2</version>
<configuration>
<port>8080</port>
<path>/</path>
</configuration>
</plugin>
</plugins>
</build>
2.1 Spring MVC 请求处理流程
流程说明
第⼀步:⽤户发送请求⾄前端控制器 DispatcherServlet
第⼆步: DispatcherServlet 收到请求调⽤ HandlerMapping 处理器映射器
第三步:处理器映射器根据请求 Url 找到具体的 Handler (后端控制器),⽣成处理器对象及处理器拦截
器 ( 如果 有则⽣成 ) ⼀并返回 DispatcherServlet
第四步: DispatcherServlet 调⽤ HandlerAdapter 处理器适配器去调⽤ Handler
第五步:处理器适配器执⾏ Handler
第六步: Handler 执⾏完成给处理器适配器返回 ModelAndView
第七步:处理器适配器向前端控制器返回 ModelAndView , ModelAndView 是 SpringMVC 框架的⼀个
底层对 象,包括 Model 和 View
第⼋步:前端控制器请求视图解析器去进⾏视图解析,根据逻辑视图名来解析真正的视图。
第九步:视图解析器向前端控制器返回 View
第⼗步:前端控制器进⾏视图渲染,就是将模型数据(在 ModelAndView 对象中)填充到 request 域
第⼗⼀步:前端控制器向⽤户响应结果
2.2 Spring MVC 九⼤组件
HandlerMapping (处理器映射器)
HandlerMapping 是⽤来查找 Handler 的,也就是处理器,具体的表现形式可以是类,也可以是
⽅法。⽐如,标注了 @RequestMapping 的每个⽅法都可以看成是⼀个 Handler 。 Handler 负责具
体实际的请求处理,在请求到达后, HandlerMapping 的作⽤便是找到请求相应的处理器
Handler 和 Interceptor.
HandlerAdapter (处理器适配器) HandlerAdapter 是⼀个适配器。因为 Spring MVC 中 Handler 可以是任意形式的,只要能处理请
求即可。但是把请求交给 Servlet 的时候,由于 Servlet 的⽅法结构都是
doService(HttpServletRequest req,HttpServletResponse resp) 形式的,要让固定的 Servlet 处理
⽅法调⽤ Handler 来进⾏处理,便是 HandlerAdapter 的职责。
HandlerExceptionResolver
HandlerExceptionResolver ⽤于处理 Handler 产⽣的异常情况。它的作⽤是根据异常设置
ModelAndView ,之后交给渲染⽅法进⾏渲染,渲染⽅法会将 ModelAndView 渲染成⻚⾯。
ViewResolver
ViewResolver 即视图解析器,⽤于将 String 类型的视图名和 Locale 解析为 View 类型的视图,只有⼀
个 resolveViewName() ⽅法。从⽅法的定义可以看出, Controller 层返回的 String 类型视图名
viewName 最终会在这⾥被解析成为 View 。 View 是⽤来渲染⻚⾯的,也就是说,它会将程序返回
的参数和数据填⼊模板中,⽣成 html ⽂件。 ViewResolver 在这个过程主要完成两件事情:
ViewResolver 找到渲染所⽤的模板(第⼀件⼤事)和所⽤的技术(第⼆件⼤事,其实也就是找到
视图的类型,如 JSP )并填⼊参数。默认情况下, Spring MVC 会⾃动为我们配置⼀个
InternalResourceViewResolver, 是针对 JSP 类型视图的。
RequestToViewNameTranslator
RequestToViewNameTranslator 组件的作⽤是从请求中获取 ViewName. 因为 ViewResolver 根据
ViewName 查找 View ,但有的 Handler 处理完成之后 , 没有设置 View ,也没有设置 ViewName ,
便要通过这个组件从请求中查找 ViewName 。
LocaleResolver
ViewResolver 组件的 resolveViewName ⽅法需要两个参数,⼀个是视图名,⼀个是 Locale 。
LocaleResolver ⽤于从请求中解析出 Locale ,⽐如中国 Locale 是 zh-CN ,⽤来表示⼀个区域。这
个组件也是 i18n 的基础。
ThemeResolver
ThemeResolver 组件是⽤来解析主题的。主题是样式、图⽚及它们所形成的显示效果的集合。
Spring MVC 中⼀套主题对应⼀个 properties ⽂件,⾥⾯存放着与当前主题相关的所有资源,如图
⽚、 CSS 样式等。创建主题⾮常简单,只需准备好资源,然后新建⼀个 “ 主题名 .properties” 并将资
源设置进去,放在 classpath 下,之后便可以在⻚⾯中使⽤了。 SpringMVC 中与主题相关的类有
ThemeResolver 、 ThemeSource 和 Theme 。 ThemeResolver 负责从请求中解析出主题名,
ThemeSource 根据主题名找到具体的主题,其抽象也就是 Theme ,可以通过 Theme 来获取主题和
具体的资源。
MultipartResolver
MultipartResolver ⽤于上传请求,通过将普通的请求包装成 MultipartHttpServletRequest 来实
现。 MultipartHttpServletRequest 可以通过 getFile() ⽅法 直接获得⽂件。如果上传多个⽂件,还
可以调⽤ getFileMap() ⽅法得到 Map<FileName , File> 这样的结构, MultipartResolver 的作⽤就
是封装普通的请求,使其拥有⽂件上传的功能。
FlashMapManager
FlashMap ⽤于重定向时的参数传递,⽐如在处理⽤户订单时候,为了避免重复提交,可以处理完
post 请求之后重定向到⼀个 get 请求,这个 get 请求可以⽤来显示订单详情之类的信息。这样做虽然
可以规避⽤户重新提交订单的问题,但是在这个⻚⾯上要显示订单的信息,这些数据从哪⾥来获得
呢?因为重定向时么有传递参数这⼀功能的,如果不想把参数写进 URL (不推荐),那么就可以通
过 FlashMap 来传递。只需要在重定向之前将要传递的数据写⼊请求(可以通过 ServletRequestAttributes.getRequest() ⽅法获得)的属性 OUTPUT_FLASH_MAP_ATTRIBUTE
中,这样在重定向之后的 Handler 中 Spring 就会⾃动将其设置到 Model 中,在显示订单信息的⻚⾯
上就可以直接从 Model 中获取数据。 FlashMapManager 就是⽤来管理 FalshMap
如何解决/拦截静态资源这件事?
<!-- 方式一:带后缀 *.action,*.do,*.aa,比较精确,方便,在以前和现在在企业中都有很大的使用比例 方式二:/ rest风格,不会拦截.jsp,但是会拦截.html等静态资源文件 (静态资源文件:除了servlet和jsp之外的js,css,.png等) 为什么配置为/会拦截静态资源, 因为tomcat中有一个web.xml(父),你的项目中有一个web.xml(子),是一个继承关系 父web.xml中有一个DefaultServlet,url-pattern 是一个/
此时我们自己的web.xml中也配置了/,复写了父web.xml的配置
为什么不拦截.jsp呢?
因为父web.xml中有一个JspServlet,这个servlet拦截.jsp文件,而我们并没有覆写这个配置,
所以springmvc此时不拦截jsp,jsp的处理交给了tomcat
如何解决/拦截静态资源这件事?
方式三:/* 拦截所有,包括jsp
-->
<!-- 匹配拦截规则的url请求,进入springmvc框架处理-->
<url-pattern>/</url-pattern>
解决方案1
解决方案2
输出机制Map、Model和ModelMap
/**
* SpringMVC在handler方法上传入Map、Model和ModelMap参数,并向这些参数中保存数据(放入到请求域),都可以在页面获取到
*
* 它们之间是什么关系?
* 运行时的具体类型都是BindingAwareModelMap,相当于给BindingAwareModelMap中保存的数据都会放在请求域中
*
* Map(jdk中的接口) Model(spring的接口)
*
* ModelMap(class,实现Map接口)
*
*
*
*
* BindingAwareModelMap继承了ExtendedModelMap,ExtendedModelMap继承了ModelMap,实现了Model接口
* 所以Map、Model和ModelMap底层都是BindingAwareModelMap
*/
/**
* 直接声明形参ModelMap,封装数据
* url: http://localhost:8080/demo/handle11
*
* =================modelmap:class org.springframework.validation.support.BindingAwareModelMap
*/
@RequestMapping("/handle11")
public String handle11(ModelMap modelMap) {
Date date = new Date();// 服务器时间
modelMap.addAttribute("date",date);
System.out.println("=================modelmap:" + modelMap.getClass());
return "success";
} /**
* 直接声明形参Model,封装数据
* url: http://localhost:8080/demo/handle12
* =================model:class org.springframework.validation.support.BindingAwareModelMap
*/
@RequestMapping("/handle12")
public String handle12(Model model) {
Date date = new Date();
model.addAttribute("date",date);
System.out.println("=================model:" + model.getClass());
return "success";
} /**
* 直接声明形参Map集合,封装数据
* url: http://localhost:8080/demo/handle13
* =================map:class org.springframework.validation.support.BindingAwareModelMap
*/
@RequestMapping("/handle13")
public String handle13(Map<String,Object> map) {
Date date = new Date();
map.put("date",date);
System.out.println("=================map:" + map.getClass());
return "success";
}