SpringMVC框架下,Web请求的流程
- 请求会首先被Spring MVC的前端请求分发器(Dispatcher)拦截。该拦截器是一个Servlet,需要在web.xml中配置,所有符合配置的URL样式的访问请求,将都会被该拦截器拦截,Spirng提供了默认的拦截器org.springframework.web.servlet.DispatcherServlet
- 在接收到访问请求后,分发器会根据Spring配置文件或代码中的注解,查找合适的控制器
- 分发器找到合适的控制器后,将请求转交给该控制器处理
- 通过控制器会调用响服务层处理业务逻辑,在请请求执行完毕后,控制器需返回处理后的数据模型Model以及视图
- 获得Spring视图后,Spring会根据视图的配置信息来显示该视图
视图和视图解析器
请求处理方法执行完成后,最终返回一个ModelAndView对象。无论处理方法返回的是String,View还是ModelMap等类型,SpringMVC都会在内部将它们装配成一个ModelAndView对象,它包含了逻辑名和模型对象的视图
Spring MVC借助视图解析器(ViewResolver)得到最终的View对象,最终的视图可以是JSP,也可能是Excel等各自表现形式的视图
视图
视图的作用是渲染模型数据,将模型数据以某种形式呈现给客户,为了实现视图模型和具体实现技术的解耦,Spring在org.springframework.web.servlet包中定义了一个高度抽象的View接口。
视图对象由视图解析器负责实例化,由于视图是无状态的,因此不存在线程安全的问题
常用视图实现类
视图类型 | 说明 |
---|---|
InternalResourceView | 将JSP或其他资源封装成一个视图,是 InternalResourceViewResolver默认使用的实现类 |
JstlView | 如果JSP文件中使用了JSTL的国际化标签的功能,则需要使用该视图类 |
AbstractExcelView | Excel文档视图的抽象类,该视图基于POL构造Excel文档 |
视图解析器
SpringMVC为逻辑视图名的解析提供了不同的策略,可以在SpringWEB上下文中配置一种或多种解析策略,并指他们之间的先后顺序。每一种映射策略对应一个具体的视图解析器实现类
视图解析器的作用就是将逻辑视图解析为一个具体的视图对象,所有的视图解析器必须实现ViewResolver接口
常用视图解析器类
视图类型 | 说明 |
---|---|
BeanNameViewResolver | 将逻辑视图解析为一个Bean,Bean的id等于逻辑视图名 |
InternalResourceViewResolver | 将视图名解析为一个URL文件,一般使用该解析器将视图名保存在WEB-INF目录下的程序文件,如JSP |
- 可以使用一种或多种视图解析器
- 每个视图解析器都实现了Ordered接口并开放出一个order属性,可以通过order属性来指定解析器的有优先顺序,order越小优先级越高
- SpringMVC会按视图解析器的优先顺序对逻辑视图名进行解析,直到解析成功返回视图对象,否则抛出异常
InternalResourceViewResolver
该视图解析器将处理器返回的结果处理为URL形式
- 若项目中使用了JSTL,则SpringMVC会自动把视图由InternalResourceView转为JstlView
- 若使用JSTL的fmt标签则需要在SpringMVC的配置文件中配置国际化资源文件
- 若希望直接相应通过SpringMVC渲染的页面,可以使用nvc:viewcontroller标签实现
自定义视图
@Component("helloView")
public class HelloView implements View{
@Override
public String getContentType() {
// TODO Auto-generated method stub
return "text/html";
}
@Override
public void render(Map<String, ?> map, HttpServletRequest request, HttpServletResponse response) throws Exception {
response.getWriter().println("this is my view");
}
}
<bean class="org.springframework.web.servlet.view.BeanNameViewResolver">
<property name="order" value="100"></property>
</bean>
@RequestMapping("/test/helloview")
public String testView() {
System.out.println("helloView");
return "helloView";
}
- 定义视图实现类,实现View接口
- 配置文件中注册BeanNameViewResolver解析器,该解析器通过控制器返回的值寻找对应的Bean,因此需要在实现View接口的实现了上标注@Componet注解
- 控制器方法返回值需要和bean注解对应,如果@Componet中没有声明属性,则默认为实现类首字母小写
重定向和转发
一般情况下,控制器方法返回字符串类型的值会被当成逻辑视图名处理
如果返回的字符串中带有**forward: 或redirect:**前缀时,SpringMVC会对他们进行特殊处理,将forward:和redirect:当成指示符,其后的字符串作为URL处理
@RequestMapping("/test/helloview")
public String testView() {
System.out.println("helloView");
//重定向到index.jsp
return "redirect:/index.jsp";
}
Spring的表单标签
通过SpringMVC的表单标签可以实现将模型数据中的属性和HTML表单元素相绑定,以实现表单数据到的更便捷编辑和表单值的回显。
通过ModelAttribute属性指定绑定的模型属性,若没有指定该属性,则默认从request域对象中读取command的表单bean,如果该属性值也不存在,则会发生错误
@RequestMapping(value="emp/{id}",method=RequestMethod.GET)
public String inputEdit(Map<String,Object> map,@PathVariable("id")Integer id) {
//放入employee进Map
map.put("employee",employeeDao.get(id));
map.put("departments",departmentDao.getDepartments());
return "input";
}
<!-- 使用springMVC的表单标签 -->
<%@ taglib prefix="form" uri="http://www.springframework.org/tags/form" %>
<!-- form表单标签绑定Map中的employee,可以在表单中直接使用Map中的对象-->
<form:form action="${pageContext.request.contextPath}/emp" method="POST" modelAttribute="employee">
<c:if test="${employee.id == null}">
LastName:<form:input path="lastName"/>
</c:if>
<c:if test="${employee.id != null}">
<form:hidden path="id"/>
<input type="hidden" name="_method" value="PUT">
</c:if>
<br>
Email:<form:input path="email"/>
<%
Map<String,String> genders = new HashMap();
genders.put("1","Male");
genders.put("0","Female");
request.setAttribute("genders",genders);
%>
Gender:<form:radiobuttons path="gender" items="${genders}"/>
<br>
Department:<form:select path="department.id" items="${departments}" itemLabel="departmentName" itemValue="id"></form:select>
<br>
Birth:<form:input path="birth"/>
<input type="submit" value="提交"/>
</form:form>
- path:表单字段,对应 html 元素的 name 属性,支持级联属性
- form:input、form:password、form:hidden、form:textarea对应 HTML 表单的 text、password、hidden、textarea标签
- form:radiobutton:单选框组件标签,当表单 bean 对应的属性值和 value 值相等时,单选框被选中
- form:radiobuttons:单选框组标签,用于构造多个单选框
- items:可以是一个 List、String[] 或 Map
- itemValue:指定 radio 的 value 值。可以是集合中 bean 的一个属性值
- itemLabel:指定 radio 的 label 值
- delimiter:多个单选框可以通过 delimiter 指定分隔符