需求:前端浏览器请求
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";
}