1入门程序
1.1需求
springmvc和mybatis用商品订单
1.2环境准备
数据库环境
mysql 5.4
mybatis
表
items orderdetail orders user
java环境
jdk1.7
eclipse indigo
spring的版本 spring3.2
将spring3.2所有jar包(一定包括spring-webmvc-3.2.0RELSASE.jar)
2.配置前端控制器
在web.xml中配置前端控制器
<!-- springmvc前端控制器 -->
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- contextConfigLocation配置springmvc加载配置文件(配置处理器映射器,适配器等等
)若果不配置,contextConfigLocation,默认加载的是WEB-INF/servlet名称-servlet.xml(springmvc-servlet.xml)
-->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<!--
第一种:*.action,访问以.action结尾的DispatcherServlet进行解析
第二种:/ ,所有访问的地址都由DispatcherServlet进行解析,对于静态文件的解析需要配置不让DispatcherServlet
进行解析,使用此方法可以实现RESTful风格的URL
第三种:/*,这样配置不对吗,使用这种配置器,最终要转发到一个jsp页面时,
仍然会由DispatcherServlet解析jsp地址,不能根据jsp页面找到handler会报错
-->
<url-pattern>*.action</url-pattern>
</servlet-mapping>
3.1.配置处理器适配器
在classpath下的springmvc..xml进行配置处理器配置
<!-- 处理器适配器
所有处理器适配器都实现HandlerAdapter接口
-->
<bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>
通过查看源代码
public class SimpleControllerHandlerAdapter implements HandlerAdapter {
@Override
public boolean supports(Object handler) {
return (handler instanceof Controller);
}
次适配器能执行实现Controller接口的Handler
public interface Controller {
/**
* Process the request and return a ModelAndView object which the DispatcherServlet
* will render. A {@code null} return value is not an error: it indicates that
* this object completed request processing itself and that there is therefore no
* ModelAndView to render.
* @param request current HTTP request
* @param response current HTTP response
* @return a ModelAndView to render, or {@code null} if handled directly
* @throws Exception in case of errors
*/
ModelAndView handleRequest(HttpServletRequest request, HttpServletResponse response) throws Exception;
}
3.2开发Handler
需要实现Controller接口,才能由org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter适配器执行
public class ItemController implements Controller{
@Override
public ModelAndView handleRequest(HttpServletRequest request,
HttpServletResponse response) throws Exception {
//商品列表
List<Items> itemsilt=new ArrayList<Items>();
Items items1=new Items();
items1.setName("联想电脑");
items1.setPrice(6000f);
items1.setDetail("Thinskkee 联想笔记本");
Items items2=new Items();
items2.setName("苹果手机");
items2.setPrice(6500f);
items2.setDetail("苹果手机");
itemsilt.add(items1);
itemsilt.add(items2);
//创建modelAndView准备填充数据,设置视图
ModelAndView modelAndView=new ModelAndView();
//相当于request的setAttribute在jsp页面中通过itemsList取数据
modelAndView.addObject("itemsList", itemsilt);
//视图
modelAndView.setViewName("/WEB-INF/jsp/items/itemsList.jsp");
return modelAndView;
}
}
3.3视图编写
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
<%@ taglib uri="http://java.sun.com/jsp/jstl/fmt" prefix="fmt" %>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
<head>
<base href="<%=basePath%>">
<title>商品列表</title>
<meta http-equiv="pragma" content="no-cache">
<meta http-equiv="cache-control" content="no-cache">
<meta http-equiv="expires" content="0">
<meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
<meta http-equiv="description" content="This is my page">
<!--
<link rel="stylesheet" type="text/css" href="styles.css">
-->
</head>
<body>
<form action="${pageContext.request.contextPath}/item/queryItem.action" method="post">
查询条件:、
<table width="100%" border=1>
<tr>
<td><input type="submit" value="查询"></td>
</tr>
</table>
<table width="100%" border=1 >
<tr>
<td>商品名称</td>
<td>商品价格</td>
<td>商品描述</td>
</tr>
<c:forEach items="${itemsList}" var="item">
<td>${item.name }</td>
<td>${item.price}</td>
<td>${item.detail }</td>
</c:forEach>
</table>
</form>
</body>
</html>
3.4配置handler
将编写的Handler在spring容器中加载
<!-- 配置handler -->
<bean name="/queryItems.action" class="cn.dxq.ssm.controller.ItemController"/>
3.5配置处理映射器、
在classpath下的springmvc.xml中配置处理器映射器
<!-- 处理器映射器
将bean的name作为URL进行查询,需要配置Handler时指定beanname(就是URL)
-->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"></bean>
3.6.配置视图解析器
部署调试
访问http://localhost/Spring_Mvc/queryItems.action
处理器映射器根据URL找不到Handler跟下边的错误,说明URL有误
HTTP Statuts 404
处理器映射器根据URL找到了Handler,转发的jsp页面找到报下边的错误,说明jsp地址错误了
4.非注解的映射器和适配器
4.1处理器映射器
org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping
4.2另一个映射器
org.springframework.web.servlet.handler.SimpleUrlHandlerMapping
<!-- 配置handler -->
<bean id="ItemController1" name="/queryItems.action" class="cn.dxq.ssm.controller.ItemController"/>
<!-- 另一个处理器映射器 -->
<!-- 简单URL映射 -->
<bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping">
<property name="mappings">
<props>
<!-- 对ItemController1进行url映射操作,URL是/queryItems1.action -->
<prop key="/queryItems1.action">ItemController1</prop>
</props>
</property>
</bean>
4.3处理器适配器
org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter
要求编写handler实现controller接口
org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter
要求编写的handler实现HttpRequestHandler接口
<!-- 另一个非注解适配器 -->
<bean class="org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter"> </bean>
public class ItemController2 implements HttpRequestHandler{
@Override
public void handleRequest(HttpServletRequest request,
HttpServletResponse response) throws ServletException, IOException {
// TODO Auto-generated method stub
//商品列表
List<Items> itemsilt=new ArrayList<Items>();
Items items1=new Items();
items1.setName("联想电脑");
items1.setPrice(6000f);
items1.setDetail("Thinskkee 联想笔记本");
Items items2=new Items();
items2.setName("苹果手机");
items2.setPrice(6500f);
items2.setDetail("苹果手机");
itemsilt.add(items1);
itemsilt.add(items2);
//设置模型数据
request.setAttribute("itemsList", itemsilt);
//设置转发视图
request.getRequestDispatcher("/WEB-INF/jsp/items/itemsList.jsp").forward(request, response);
}
//使用此类方法可以通过修改response,设置响应的数据格式,比如响应json数据
response.setCharacterEncoding("utf-8");
response.setContextType("application/json;charset=utf-8");
respomse.getWriter().writer("json串");
5.DispatcherServlet.properties
/org/springframework/web/servlet/DispatcherServlet.properties
前端控制器从上边的文件中加载处理器映射器,适配器,视图解析器等组件,如果不在springmvc.xml中配置,使用默认加载的
6.注解的映射器和适配器
在spring3.1之前使用的是org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping注解映射器
之后是org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping注解映射器
在spring3.1之前使用的适配器是org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter注解适配器
之后是org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter注解适配器
<!-- 注解映射器 -->
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"></bean>
<!-- 注解适配器 -->
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"></bean>
<!-- 使用mvc:annotation-driven代替上边的注解映射器和适配器配置
mvc:annotation-driven默认加载很多的参数绑定方法
比如json转换器就默认加载了,如果使用mvc:annotation-driven不用配置上边的
RequestMappingHandlerMapping和RequestMappingHandlerAdapter
-->
<!-- <mvc:annotation-driven></mvc:annotation-driven> -->
5.2开发注解的handler
使用注解的映射器和注解适配器,(注解的映射器和注解的适配器必须配对使用)
//使用@Controller表示它是一个控制器
@Controller
public class ItemController3 {
//商品查询列表
/**
* @RequestMapping实现对queryItems方法和URL进行映射,一个方法对应一个URL
*一般建议方法和URL名字一致
*/
@RequestMapping("/queryItem_1.action")
public ModelAndView queryItenm() throws Exception{
//商品列表
List<Items> itemsilt=new ArrayList<Items>();
Items items1=new Items();
items1.setName("联想电脑");
items1.setPrice(6000f);
items1.setDetail("Thinskkee 联想笔记本");
Items items2=new Items();
items2.setName("苹果手机");
items2.setPrice(6500f);
items2.setDetail("苹果手机");
itemsilt.add(items1);
itemsilt.add(items2);
//创建modelAndView准备填充数据,设置视图
ModelAndView modelAndView=new ModelAndView();
//相当于request的setAttribute在jsp页面中通过itemsList取数据
modelAndView.addObject("itemsList", itemsilt);
//视图
modelAndView.setViewName("/WEB-INF/jsp/items/itemsList.jsp");
return modelAndView;
}
5.3在spring容器中加载handler
<!-- 对于注解的handler可以单个配置
实际开发中建议使用组件扫描
-->
<!-- <bean class="cn.dxq.ssm.controller.ItemController3"></bean> -->
<!-- 可以扫描controller service
让它们扫描 controller 指定的controller包
-->
<context:component-scan base-package="cn.dxq.ssm.controller"></context:component-scan>
6.源码分析
通过前端控制器源码分析springmvc的执行过程
第一步:前端控制器接受请求
调用doDispatch
protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception {
HttpServletRequest processedRequest = request;
HandlerExecutionChain mappedHandler = null;
boolean multipartRequestParsed = false;
第二步:前端控制器调用处理器映射器差查找handler
protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
for (HandlerMapping hm : this.handlerMappings) {
if (logger.isTraceEnabled()) {
logger.trace(
"Testing handler map [" + hm + "] in DispatcherServlet with name '" + getServletName() + "'");
}
HandlerExecutionChain handler = hm.getHandler(request);
if (handler != null) {
return handler;
}
}
return null;
}
第三步:调用处理器适配器执行handler,得到执行结果ModelAndView
mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
第四步:视图渲染:将model数据填充到request域
视图解析 得到view
View=resolveViewName(mv.getViewName(),mv.getModelInternal(). locale, request);
调用view渲染方法,将model数据填充到request域
view.render(mv.getModelInternal(),request,response);
7.入门程序小结
理解springmvc前端控制器,处理器映射器,处理器适配器,视图解析器用法
前端控制器:配置URL
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<!--
第一种:*.action,访问以.action结尾的DispatcherServlet进行解析
第二种:/ ,所有访问的地址都由DispatcherServlet进行解析,对于静态文件的解析需要配置不让DispatcherServlet
进行解析,使用此方法可以实现RESTful风格的URL
第三种:/*,这样配置不对吗,使用这种配置器,最终要转发到一个jsp页面时,
仍然会由DispatcherServlet解析jsp地址,不能根据jsp页面找到handler会报错
-->
<url-pattern>*.action</url-pattern>
</servlet-mapping>
处理器映射器
非注解的处理器映射器(了解)
注解的处理器映射器(掌握)
对标记有@controller类中标记有@RequestMapping的方法进行映射,在@RequestMapping里边定义映射的URL
使用注解的映射器,不用在xml中配置URL和handler的映射关系
处理器适配器
非注解的处理器适配器、(了解)
注解的处理器适配器(掌握)
注解处理器适配器和注解处理器映射器是配对使用,理解为不能使用非注解映射器进行映射
<!-- <mvc:annotation-driven></mvc:annotation-driven> --> 可以代替下边的配置
<!-- 注解映射器 -->
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"></bean>
<!-- 注解适配器 -->
<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"></bean>
实际开发用
mvc:annotation-driven
//下面的路径,如果在视图解析器中配置jsp路径的前缀和jsp路径的后缀,修改为
//modelAndView .setViewName("/WEB-INF/jsp/items/itemsList.jsp")
//上边的路径配置可以不再程序2总指定jsp路径的前缀和jsp的后缀
modelAndView.setViewName("items/itemsList");
<!-- 视图解析器
解析jsp。默认使用jstl标签classpath下的有jstl包
-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<property name="prefix" value="/WEB-INF/jsp/"></property>
<property name="suffix" value=".jsp"></property>
</bean>