软件开发中的MVC设计模式

  软件开发的目标是减小耦合,让模块之前关系清晰.

  MVC模式在软件开发中经常和ORM模式一起应用,主要作用是将(数据抽象,数据实体传输和前台数据展示)分层,这样前台,后台,数据库三层分离,减小耦合.

  

  1)Model  :  负责抽象并存储数据库数据

  2)Controller :  负责数据的转化和传输

  3)View    :  负责展示数据

  注意 框架的应用使软件开发变得更有章可循,更规范化,软件开发的每个职责都落到了具体的模块中去

      ORM框架 : 将数据库数据封装存储至Model类中

      MVC框架 : 前台数据与实体数据的互为转化

      开发者   : 配置框架,编写业务代码,前台数据展示页面

SpringMVC工作流

  SpringMVC是建立在Spring framework之上的一个MVC框架,主要的组件有前端控制器处理器映射器处理器适配器视图解析器.

   

  1.SpringMVC的原理就如上图所展示的,客户端发来HTTP请求,

  2.前端控制器作为一个顶层执行者获取请求,

  3.返回一个执行链HandlerExecutionChain{HandlerInterceptor1,HandlerInterceptor2,Handler},

  4.前端控制器请求映射器查询对应的处理器适配器,

  5.处理器适配器代理执行处理器,

  6.处理器响应处理,返回一个ModelAndView,

  7.适配器返回一个ModelAndView至前端控制器

  8.前端控制器请求视图解析器执行解析,

  9.解析器返回一个未经渲染的视图

  10.前端控制器渲染视图,将模型数据存于request域

  11.视图发至浏览器进行渲染呈现

  注意,上图分为Model,View,Controller等模块,其中Model里面可封装许多逻辑,这一部分未在步骤上进行说明.

对比Struts2

  1.SpringMVC的前端入口是一个名为DispatchServlet的servlet,Struts2是一个Filter过滤器;

  2.SpringMVC基于方法进行开发(一个url对应一个方法),请求参数传递到方法的形参,可以设计为单例或者多例模式,Struts2是基于类开发,传递的参数通过类的属性进行传递;

  3.Struts2采用值栈存储请求和响应,通过OGNL存取数据,SpringMVC通过参数解析器将request内容进行解析,并给方法形参赋值,将数据和视图封装成一个ModelAndView对象,最后将ModelAndView的数据传入request中以及视图渲染,JSP使用JSTL来进行取值操作;

组件职责描述

1.前端控制器DispatcherServlet(无需开发者开发)

作用 : 接收请求,响应结果,相当于转发器,中央处理器.

  有了DispatcherServlet减少了其它组件之间的耦合度.

类位置  org.springframework.web.servlet.DispatcherServlet.class

配置文件  /org/springframework/web/servlet/DispatcherServlet.properties

<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>*.action</url-pattern>
</servlet-mapping>

配置方式

2.处理器映射器HandlerMapping(无需开发者开发)

    作用:根据请求的url查找Handler和Interceptor拦截器,

      将它们封装在HandlerExecutionChain 对象中给前端控制器返回.

类位置  org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping.class

<!—beanName Url映射器 -->
<bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping"/>

配置方式

        org.springframework.web.servlet.handler.SimpleUrlHandlerMapping.class

<!—简单url映射
simpleUrlHandlerMapping是BeanNameUrlHandlerMapping的增强版本,它可以将url和处理器bean的id进行统一映射配置--><bean class="org.springframework.web.servlet.handler.SimpleUrlHandlerMapping"><property name="mappings"><props><prop key="/items1.action">controller的bean id</prop><prop key="/items2.action">controller的bean id</prop></props></property></bean>

配置方式

     org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping.class

<bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/>

配置方式

3.处理器适配器HandlerAdapter(无需开发者开发)

作用:按照特定规则(HandlerAdapter要求的规则)去执行Handler

    类位置  org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter.class

<!--
所有实现了org.springframework.web.servlet.mvc.Controller 接口的Bean通过此适配器进行适配、执行.--><bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter" />

配置方式

         org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter.class

<!--
所有实现了org.springframework.web.HttpRequestHandler 接口的Bean通过此适配器进行适配、执行.
--><bean class="org.springframework.web.servlet.mvc.HttpRequestHandlerAdapter"/>

配置方式

         org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter.class

<!--注解时使用的Adapter,配合注解映射器一起使用--><bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/>

配置方式

4.处理器Handler(需开发者开发)

注意:编写Handler时按照HandlerAdapter的要求去做,这样适配器才可以去正确执行Handler

    非注解方式

package com.harry.controller;import java.util.ArrayList;
import java.util.List;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;import com.harry.entity.Items;/**** @author harry*    用于处理页面请求,*    数据来源是通过数据库ORM映射至实体类来获取,*    这里使用静态获取数据进行简单演示.*/
public class ItemList_1 implements Controller {@Overridepublic ModelAndView handleRequest(HttpServletRequest request,HttpServletResponse response) throws Exception {//调用service查找 数据库,查询商品列表,这里使用静态数据模拟List<Items> itemsList = new ArrayList<Items>();//向list中填充静态数据
        Items items_1 = new Items();items_1.setName("联想笔记本");items_1.setPrice(6000f);items_1.setDetail("ThinkPad T430 联想笔记本电脑!");Items items_2 = new Items();items_2.setName("苹果手机");items_2.setPrice(5000f);items_2.setDetail("iphone6苹果手机!");itemsList.add(items_1);itemsList.add(items_2);//返回ModelAndViewModelAndView modelAndView =  new ModelAndView();//相当 于request的setAttribut,在jsp页面中通过itemsList取数据modelAndView.addObject("itemsList", itemsList);//指定视图modelAndView.setViewName("Items/itemList");return modelAndView;}}

ItemList_1.java

    注解方式

@Controller
public class ItemList_2 {
/*RequestMappingHandlerMapping将对类中标记@ResquestMapping的方法进行映射,根据ResquestMapping定义的url匹配ResquestMapping标记的方法,匹配成功返回HandlerMethod对象给前端控制器,HandlerMethod对象中封装url对应的方法Method
*/@RequestMapping("/queryItem.action")public ModelAndView queryItem() {// 商品列表List<Items> itemsList = new ArrayList<Items>();Items items_1 = new Items();items_1.setName("联想笔记本");items_1.setPrice(6000f);items_1.setDetail("ThinkPad T430 联想笔记本电脑!");Items items_2 = new Items();items_2.setName("苹果手机");items_2.setPrice(5000f);items_2.setDetail("iphone6苹果手机!");itemsList.add(items_1);itemsList.add(items_2);// 创建modelAndView准备填充数据、设置视图ModelAndView modelAndView = new ModelAndView();// 填充数据modelAndView.addObject("itemsList", itemsList);// 视图modelAndView.setViewName("order/itemsList");return modelAndView;}}

ItemList_2.java

注意,注解方式应使用注解包扫描<context:component-scan base-package="com.harry.controller"/>,

包扫描可以扫描到@Bean,@Controller,@Service,@Repository等注解类.

RequestMappingHandlerMapping,HttpRequestHandlerAdapter(<mvc:annotation-driven>可自动加载这两个类)

5.视图解析器View resolver(无需开发者开发)

作用:进行视图解析,根据逻辑视图名解析成真正的视图(view)

类位置  org.springframework.web.servlet.view.InternalResourceViewResolver.class

<!-- ViewResolver配置 -->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/><property name="prefix" value="/WEB-INF/jsp/"/><property name="suffix" value=".jsp"/>
</bean>

配置方式

6.视图View(需开发者开发)

View是一个接口,实现类支持不同的View类型(jsp、freemarker、pdf...) 

案例驱动开发

  springmvc的开发,需要配置上述工作流中的组件,入门程序紧接着框架应用:Mybatis - 开发详述的案例进行开发,

  1.备份数据库,实体类

-- MySQL dump 10.13  Distrib 5.7.18, for Linux (x86_64)
--
-- Host: localhost    Database: mybatis
-- ------------------------------------------------------
-- Server version    5.7.18-0ubuntu0.16.04.1/*!40101 SET @OLD_CHARACTER_SET_CLIENT=@@CHARACTER_SET_CLIENT */;
/*!40101 SET @OLD_CHARACTER_SET_RESULTS=@@CHARACTER_SET_RESULTS */;
/*!40101 SET @OLD_COLLATION_CONNECTION=@@COLLATION_CONNECTION */;
/*!40101 SET NAMES utf8 */;
/*!40103 SET @OLD_TIME_ZONE=@@TIME_ZONE */;
/*!40103 SET TIME_ZONE='+00:00' */;
/*!40014 SET @OLD_UNIQUE_CHECKS=@@UNIQUE_CHECKS, UNIQUE_CHECKS=0 */;
/*!40014 SET @OLD_FOREIGN_KEY_CHECKS=@@FOREIGN_KEY_CHECKS, FOREIGN_KEY_CHECKS=0 */;
/*!40101 SET @OLD_SQL_MODE=@@SQL_MODE, SQL_MODE='NO_AUTO_VALUE_ON_ZERO' */;
/*!40111 SET @OLD_SQL_NOTES=@@SQL_NOTES, SQL_NOTES=0 */;--
-- Table structure for table `items`
--

DROP TABLE IF EXISTS `items`;
/*!40101 SET @saved_cs_client     = @@character_set_client */;
/*!40101 SET character_set_client = utf8 */;
CREATE TABLE `items` (`id` int(11) NOT NULL AUTO_INCREMENT,`name` varchar(32) NOT NULL COMMENT '商品名称',`price` float(10,1) NOT NULL COMMENT '商品定价',`detail` text COMMENT '商品描述',`pic` varchar(64) DEFAULT NULL COMMENT '商品图片',`createtime` datetime NOT NULL COMMENT '生产日期',PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=4 DEFAULT CHARSET=utf8;
/*!40101 SET character_set_client = @saved_cs_client */;--
-- Dumping data for table `items`
--

LOCK TABLES `items` WRITE;
/*!40000 ALTER TABLE `items` DISABLE KEYS */;
INSERT INTO `items` VALUES (1,'台式机',3000.0,'该电脑质量非常好!!!!',NULL,'2015-02-03 13:22:53'),(2,'笔记本',6000.0,'笔记本性能好,质量好!!!!!',NULL,'2015-02-09 13:22:57'),(3,'背包',200.0,'名牌背包,容量大质量好!!!!',NULL,'2015-02-06 13:23:02');
/*!40000 ALTER TABLE `items` ENABLE KEYS */;
UNLOCK TABLES;--
-- Table structure for table `orderdetail`
--

DROP TABLE IF EXISTS `orderdetail`;
/*!40101 SET @saved_cs_client     = @@character_set_client */;
/*!40101 SET character_set_client = utf8 */;
CREATE TABLE `orderdetail` (`id` int(11) NOT NULL AUTO_INCREMENT,`orders_id` int(11) NOT NULL COMMENT '订单id',`items_id` int(11) NOT NULL COMMENT '商品id',`items_num` int(11) DEFAULT NULL COMMENT '商品购买数量',PRIMARY KEY (`id`),KEY `FK_orderdetail_1` (`orders_id`),KEY `FK_orderdetail_2` (`items_id`),CONSTRAINT `FK_orderdetail_1` FOREIGN KEY (`orders_id`) REFERENCES `orders` (`id`) ON DELETE NO ACTION ON UPDATE NO ACTION,CONSTRAINT `FK_orderdetail_2` FOREIGN KEY (`items_id`) REFERENCES `items` (`id`) ON DELETE NO ACTION ON UPDATE NO ACTION
) ENGINE=InnoDB AUTO_INCREMENT=5 DEFAULT CHARSET=utf8;
/*!40101 SET character_set_client = @saved_cs_client */;--
-- Dumping data for table `orderdetail`
--

LOCK TABLES `orderdetail` WRITE;
/*!40000 ALTER TABLE `orderdetail` DISABLE KEYS */;
INSERT INTO `orderdetail` VALUES (1,3,1,1),(2,3,2,3),(3,4,3,4),(4,4,2,3);
/*!40000 ALTER TABLE `orderdetail` ENABLE KEYS */;
UNLOCK TABLES;--
-- Table structure for table `orders`
--

DROP TABLE IF EXISTS `orders`;
/*!40101 SET @saved_cs_client     = @@character_set_client */;
/*!40101 SET character_set_client = utf8 */;
CREATE TABLE `orders` (`id` int(11) NOT NULL AUTO_INCREMENT,`user_id` int(11) NOT NULL COMMENT '下单用户id',`number` varchar(32) NOT NULL COMMENT '订单号',`createtime` datetime NOT NULL COMMENT '创建订单时间',`note` varchar(100) DEFAULT NULL COMMENT '备注',PRIMARY KEY (`id`),KEY `FK_orders_1` (`user_id`),CONSTRAINT `FK_orders_id` FOREIGN KEY (`user_id`) REFERENCES `user` (`id`) ON DELETE NO ACTION ON UPDATE NO ACTION
) ENGINE=InnoDB AUTO_INCREMENT=6 DEFAULT CHARSET=utf8;
/*!40101 SET character_set_client = @saved_cs_client */;--
-- Dumping data for table `orders`
--

LOCK TABLES `orders` WRITE;
/*!40000 ALTER TABLE `orders` DISABLE KEYS */;
INSERT INTO `orders` VALUES (3,1,'1000010','2015-02-04 13:22:35',NULL),(4,1,'1000011','2015-02-03 13:22:41',NULL),(5,10,'1000012','2015-02-12 16:13:23',NULL);
/*!40000 ALTER TABLE `orders` ENABLE KEYS */;
UNLOCK TABLES;--
-- Table structure for table `user`
--

DROP TABLE IF EXISTS `user`;
/*!40101 SET @saved_cs_client     = @@character_set_client */;
/*!40101 SET character_set_client = utf8 */;
CREATE TABLE `user` (`id` int(11) NOT NULL AUTO_INCREMENT,`username` varchar(32) NOT NULL COMMENT '用户名称',`birthday` date DEFAULT NULL COMMENT '生日',`sex` char(1) DEFAULT NULL COMMENT '性别',`address` varchar(256) DEFAULT NULL COMMENT '地址',PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=27 DEFAULT CHARSET=utf8;
/*!40101 SET character_set_client = @saved_cs_client */;--
-- Dumping data for table `user`
--

LOCK TABLES `user` WRITE;
/*!40000 ALTER TABLE `user` DISABLE KEYS */;
INSERT INTO `user` VALUES (1,'王五',NULL,'2',NULL),(10,'张三','2014-07-10','1','北京市'),(16,'张小明',NULL,'1','河南郑州'),(22,'陈小明',NULL,'1','河南郑州'),(24,'张三丰',NULL,'1','河南郑州'),(25,'陈小明',NULL,'1','河南郑州'),(26,'王五',NULL,NULL,NULL);
/*!40000 ALTER TABLE `user` ENABLE KEYS */;
UNLOCK TABLES;
/*!40103 SET TIME_ZONE=@OLD_TIME_ZONE */;/*!40101 SET SQL_MODE=@OLD_SQL_MODE */;
/*!40014 SET FOREIGN_KEY_CHECKS=@OLD_FOREIGN_KEY_CHECKS */;
/*!40014 SET UNIQUE_CHECKS=@OLD_UNIQUE_CHECKS */;
/*!40101 SET CHARACTER_SET_CLIENT=@OLD_CHARACTER_SET_CLIENT */;
/*!40101 SET CHARACTER_SET_RESULTS=@OLD_CHARACTER_SET_RESULTS */;
/*!40101 SET COLLATION_CONNECTION=@OLD_COLLATION_CONNECTION */;
/*!40111 SET SQL_NOTES=@OLD_SQL_NOTES */;-- Dump completed on 2017-10-10  0:00:04

sql

package com.harry.entity;import java.util.Date;public class Items {private Integer id;private String name;private Float price;private String pic;private Date createtime;private String detail;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name == null ? null : name.trim();}public Float getPrice() {return price;}public void setPrice(Float price) {this.price = price;}public String getPic() {return pic;}public void setPic(String pic) {this.pic = pic == null ? null : pic.trim();}public Date getCreatetime() {return createtime;}public void setCreatetime(Date createtime) {this.createtime = createtime;}public String getDetail() {return detail;}public void setDetail(String detail) {this.detail = detail == null ? null : detail.trim();}
}

Items.java

package com.harry.entity;import java.util.Date;
import java.util.List;public class Orders {private Integer id;private Integer userId;private String number;private Date createtime;private String note;//用户信息private User user;//订单明细private List<Orderdetail> orderdetails;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public Integer getUserId() {return userId;}public void setUserId(Integer userId) {this.userId = userId;}public String getNumber() {return number;}public void setNumber(String number) {this.number = number == null ? null : number.trim();}public Date getCreatetime() {return createtime;}public void setCreatetime(Date createtime) {this.createtime = createtime;}public String getNote() {return note;}public void setNote(String note) {this.note = note == null ? null : note.trim();}public User getUser() {return user;}public void setUser(User user) {this.user = user;}public List<Orderdetail> getOrderdetails() {return orderdetails;}public void setOrderdetails(List<Orderdetail> orderdetails) {this.orderdetails = orderdetails;}@Overridepublic String toString() {return "Orders [id=" + id + ", userId=" + userId + ", number=" + number + ", createtime=" + createtime+ ", note=" + note + ", user=" + user + ", orderdetails=" + orderdetails + "]";}}

Orders.java

package com.harry.entity;public class Orderdetail {private Integer id;private Integer ordersId;private Integer itemsId;private Integer itemsNum;//明细对应的商品信息private Items items;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public Integer getOrdersId() {return ordersId;}public void setOrdersId(Integer ordersId) {this.ordersId = ordersId;}public Integer getItemsId() {return itemsId;}public void setItemsId(Integer itemsId) {this.itemsId = itemsId;}public Integer getItemsNum() {return itemsNum;}public void setItemsNum(Integer itemsNum) {this.itemsNum = itemsNum;}public Items getItems() {return items;}public void setItems(Items items) {this.items = items;}@Overridepublic String toString() {return "Orderdetail [id=" + id + ", ordersId=" + ordersId+ ", itemsId=" + itemsId + ", itemsNum=" + itemsNum + "]";}}

Orderdetail.java

package com.harry.entity;import java.io.Serializable;
import java.util.Date;
import java.util.List;@SuppressWarnings("serial")
public class User implements Serializable {//属性名和数据库表的字段对应private int id;private String username;// 用户姓名private String sex;// 性别private Date birthday;// 生日private String address;// 地址//用户创建的订单列表private List<Orders> ordersList;public int getId() {return id;}public void setId(int id) {this.id = id;}public String getUsername() {return username;}public void setUsername(String username) {this.username = username;}public String getSex() {return sex;}public void setSex(String sex) {this.sex = sex;}public Date getBirthday() {return birthday;}public void setBirthday(Date birthday) {this.birthday = birthday;}public String getAddress() {return address;}public void setAddress(String address) {this.address = address;}@Overridepublic String toString() {return "User [id=" + id + ", username=" + username + ", sex=" + sex+ ", birthday=" + birthday + ", address=" + address + "]";}public List<Orders> getOrdersList() {return ordersList;}public void setOrdersList(List<Orders> ordersList) {this.ordersList = ordersList;}}

User.java

  2.导入spring ioc, aop, mvc, log4j...的jar包,这里使用maven 

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsdhttp://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsdhttp://www.springframework.org/schema/cache http://www.springframework.org/schema/jdbc/spring-cache.xsd"><modelVersion>4.0.0</modelVersion><groupId>com.harry</groupId><artifactId>harry-ssm</artifactId><packaging>war</packaging><version>0.0.1-SNAPSHOT</version><name>harry-ssm Maven Webapp</name><url>http://maven.apache.org</url><dependencies><!-- https://mvnrepository.com/artifact/org.mybatis/mybatis -->
<dependency><groupId>org.mybatis</groupId><artifactId>mybatis</artifactId><version>3.4.1</version>
</dependency>
<!-- https://mvnrepository.com/artifact/log4j/log4j -->
<dependency><groupId>log4j</groupId><artifactId>log4j</artifactId><version>1.2.17</version>
</dependency><!-- https://mvnrepository.com/artifact/org.springframework/spring-core -->
<dependency><groupId>org.springframework</groupId><artifactId>spring-core</artifactId><version>4.3.11.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
<dependency><groupId>org.springframework</groupId><artifactId>spring-aop</artifactId><version>4.3.11.RELEASE</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
<dependency><groupId>org.springframework</groupId><artifactId>spring-webmvc</artifactId><version>4.3.11.RELEASE</version>
</dependency><!-- https://mvnrepository.com/artifact/junit/junit -->
<dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version><scope>test</scope>
</dependency>
<!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
<dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.40</version>
</dependency><!-- https://mvnrepository.com/artifact/taglibs/standard -->
<dependency><groupId>taglibs</groupId><artifactId>standard</artifactId><version>1.1.2</version>
</dependency><!-- https://mvnrepository.com/artifact/javax.servlet/jstl -->
<dependency><groupId>javax.servlet</groupId><artifactId>jstl</artifactId><version>1.2</version>
</dependency></dependencies><build><finalName>harry-ssm</finalName></build>
</project>

pom.xml  

  3.配置前端控制器,springmvc的入口组件

<!--
contextConfigLocation指定springmvc的主配置文件地址,如不指定则默认为WEB-INF/[DispatcherServlet 的Servlet 名字]-servlet.xml<load-on-startup>1</load-on-startup> 服务器启动时创建<url-pattern>*.action</url-pattern> 拦截地址指明所有.action结尾的地址--><servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>*.action</url-pattern>
</servlet-mapping>

web.xml

  4.配置处理器适配器

<!--
配置处理器适配器:简单处理器适配器
所有实现了org.springframework.web.servlet.mvc.Controller 接口的Bean作为Springmvc的后端控制器。
-->
<bean        class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter"/>

springmvc.xml

  5.配置处理器

package com.harry.controller;import java.util.ArrayList;
import java.util.List;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.mvc.Controller;import com.harry.entity.Items;/**** @author harry*    用于处理页面请求,*    数据来源是通过数据库ORM映射至实体类来获取,*    这里使用静态获取数据进行简单演示.*/
public class ItemList_1 implements Controller {@Overridepublic ModelAndView handleRequest(HttpServletRequest request,HttpServletResponse response) throws Exception {//调用service查找 数据库,查询商品列表,这里使用静态数据模拟List<Items> itemsList = new ArrayList<Items>();//向list中填充静态数据Items items_1 = new Items();items_1.setName("联想笔记本");items_1.setPrice(6000f);items_1.setDetail("ThinkPad T430 联想笔记本电脑!");Items items_2 = new Items();items_2.setName("苹果手机");items_2.setPrice(5000f);items_2.setDetail("iphone6苹果手机!");itemsList.add(items_1);itemsList.add(items_2);//返回ModelAndViewModelAndView modelAndView =  new ModelAndView();//相当 于request的setAttribut,在jsp页面中通过itemsList取数据modelAndView.addObject("itemsList", itemsList);//指定视图modelAndView.setViewName("Items/itemList");return modelAndView;}}

ItemList_1.java

<!-- controller配置 --><bean name="/items1.action" id="itemList1"
class="cn.itcast.springmvc.controller.first.ItemList1"/>

springmvc.xml

  6.配置处理器映射器

<!-- 处理器映射器 --><!-- 根据bean的name进行查找Handler 将action的url配置在bean的name中 --><beanclass="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping" />

springmvc.xml

  7.配置视图解析器

<!--
ViewResolver配置
JstlView 指视图使用jstl类库支持
prefix,前缀
suffix,后缀
-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="viewClass" value="org.springframework.web.servlet.view.JstlView"/><property name="prefix" value="/WEB-INF/jsp/"/><property name="suffix" value=".jsp"/>
</bean>

springmvc.xml

  8.编写视图

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%>
<%@ page isELIgnored ="false" %>
<%@ 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" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>查询商品列表</title>
</head>
<body>
商品列表:
<table width="100%" border=1>
<tr><td>商品名称</td><td>商品价格</td><td>商品描述</td>
</tr>
<c:forEach items="${itemsList }" var="item">
<tr><td>${item.name}</td><td>${item.price}</td><td>${item.detail}</td>
</tr>
</c:forEach></table>
</body>
</html>

itemList.jsp

  9.访问http://localhost:8080/harry-ssm/items1.action

    

源码工作流

  描述的工作流与上面画的图一致,只不过这次使用源码来进行流程分析.

  1.读取context中的配置,初始化各大组件

  

  2.客户端发来请求,DispatchServlet接收请求,触发doService方法,doService方法调用doDispatch方法

  

  

  3.doDispatch调用getHandler获取Handler,调用getHandlerAdapter获取Adapter

  

  

  4.getHandler通过HandlerMapping对象返回HandlerExecutionChain

  

  5.从HandlerExecutionChain中获取Handler并交由HandlerAdapter执行

  

  6.获取的ModelAndView对象,调用render方法渲染  

  

  7.render方法通过调用视图解析器来获取视图

  

  

  8.DispatchServlet渲染视图,将Model数据放在request域中

  

  9.客户端读取响应

整合SSM

  Spring整合Mybatis这个ORM框架已经在 框架应用:Mybatis  - 开发详述 整合过了,现在三个框架进行整合.

  需要完成的工作其实就是处理这关系线:

    User <--> View <--> Action <--> Service <--> DAO <--> DB

                ---------------------Spring-------------------

  DAO层连接数据库需要要数据源信息db.properties,框架需要log4j.properties,mybatis的总配置文件sqlMapConfig.xml,使用spring来配置数据源,管理事务,配置SqlSessionFactory和mapper扫描器的applicationContext-dao.xml,Mapper对象的配置文件XXXMapper.xml.

  Service层用spring配置service接口applicationContext-service.xml,配置事务管理applicationContext-transaction.xml,Service接口 

  Action层springmvc配置文件springmvc.xml,web.xml中配置DispatchServlet,配置编码转换器,加载spring容器,Controller类

    

    DAO

jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/mybatis
jdbc.username=XXXX
jdbc.password=XXXX

db.properties

# Global logging configuration,建议开发环境中要用debug
log4j.rootLogger=DEBUG, stdout
# Console output...
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%5p [%t] - %m%n

log4j.properties

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration><!—使用自动扫描器时,mapper.xml文件如果和mapper.java接口在一个目录则此处不用定义mappers -->
<mappers>
<package name="com.harry.ssm.mapper" />
</mappers>
</configuration>

sqlMapConfig.xml

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd ">
<!-- 加载配置文件 -->
<context:property-placeholder location="classpath:db.properties"/>
<!-- 数据库连接池 -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource" destroy-method="close"><property name="driverClassName" value="${jdbc.driver}"/><property name="url" value="${jdbc.url}"/><property name="username" value="${jdbc.username}"/><property name="password" value="${jdbc.password}"/><property name="maxActive" value="30"/><property name="maxIdle" value="5"/>
</bean>    <!-- 让spring管理sqlsessionfactory 使用mybatis和spring整合包中的 --><bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean"><!-- 数据库连接池 --><property name="dataSource" ref="dataSource" /><!-- 加载mybatis的全局配置文件 --><property name="configLocation" value="classpath:mybatis/SqlMapConfig.xml" /></bean>
<!-- mapper扫描器 -->
<bean class="org.mybatis.spring.mapper.MapperScannerConfigurer"><property name="basePackage" value="com.harry.ssm.mapper"></property>
<property name="sqlSessionFactoryBeanName" value="sqlSessionFactory"/></bean></beans>

applicationContext-dao.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.harry.ssm.mapper.ItemsMapper"><!-- sql片段 --><!-- 商品查询条件 --><sql id="query_items_where"><if test="items!=null"><if test="items.name!=null and items.name!=''">and items.name like '%${items.name}%'</if></if></sql><!-- 查询商品信息 --><select id="findItemsList" parameterType="queryVo" resultType="items">select * from items <where><include refid="query_items_where"/></where></select></mapper>

ItemsMapper.xml

public interface ItemsMapper {//商品列表public List<Items> findItemsList(QueryVo queryVo) throws Exception;
}

ItemsMapper.java

    Service

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd ">
<!-- 商品管理的service -->
<bean id="itemsService" class="com.harry.ssm.service.impl.ItemsServiceImpl"/>
</beans>

applicationContext-service.xml

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd "><!-- 事务管理器 对mybatis操作数据库事务控制,spring使用jdbc的事务控制类
-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"><!-- 数据源dataSource在applicationContext-dao.xml中配置了--><property name="dataSource" ref="dataSource"/>
</bean><!-- 通知 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager"><tx:attributes><!-- 传播行为 --><tx:method name="save*" propagation="REQUIRED"/><tx:method name="delete*" propagation="REQUIRED"/><tx:method name="insert*" propagation="REQUIRED"/><tx:method name="update*" propagation="REQUIRED"/><tx:method name="find*" propagation="SUPPORTS" read-only="true"/><tx:method name="get*" propagation="SUPPORTS" read-only="true"/><tx:method name="select*" propagation="SUPPORTS" read-only="true"/></tx:attributes>
</tx:advice>
<!-- aop -->
<aop:config><aop:advisor advice-ref="txAdvice" pointcut="execution(* com.harry.ssm.service.impl.*.*(..))"/>
</aop:config></beans>

applicationContext-transaction.xml

package com.harry.ssm.service;import java.util.List;import com.harryt.ssm.po.ItemsCustom;
import com.harry.ssm.po.ItemsQueryVo;public interface ItemsService {//商品查询列表public List<ItemsCustom> findItemsList(ItemsQueryVo itemsQueryVo) throws Exception;//根据id查询商品信息/*** * <p>Title: findItemsById</p>* <p>Description: </p>* @param id 查询商品的id* @return* @throws Exception*/public ItemsCustom findItemsById(Integer id) throws Exception;//修改商品信息/*** * <p>Title: updateItems</p>* <p>Description: </p>* @param id 修改商品的id* @param itemsCustom 修改的商品信息* @throws Exception*/public void updateItems(Integer id,ItemsCustom itemsCustom) throws Exception;}

IItemsService.java

package com.harry.ssm.service.impl;import java.util.List;import org.springframework.beans.BeanUtils;
import org.springframework.beans.factory.annotation.Autowired;import com.harry.ssm.mapper.ItemsMapper;
import com.harry.ssm.mapper.ItemsMapperCustom;
import com.harry.ssm.po.Items;
import com.harry.ssm.po.ItemsCustom;
import com.harry.ssm.po.ItemsQueryVo;
import com.harry.ssm.service.ItemsService;public class ItemsServiceImpl implements ItemsService{@Autowiredprivate ItemsMapperCustom itemsMapperCustom;@Autowiredprivate ItemsMapper itemsMapper;@Overridepublic List<ItemsCustom> findItemsList(ItemsQueryVo itemsQueryVo)throws Exception {//通过ItemsMapperCustom查询数据库return itemsMapperCustom.findItemsList(itemsQueryVo);}@Overridepublic ItemsCustom findItemsById(Integer id) throws Exception {Items items = itemsMapper.selectByPrimaryKey(id);//中间对商品信息进行业务处理//....//返回ItemsCustomItemsCustom itemsCustom = new ItemsCustom();//将items的属性值拷贝到itemsCustom
        BeanUtils.copyProperties(items, itemsCustom);return itemsCustom;}@Overridepublic void updateItems(Integer id, ItemsCustom itemsCustom) throws Exception {//添加业务校验,通常在service接口对关键参数进行校验//校验 id是否为空,如果为空抛出异常//更新商品信息使用updateByPrimaryKeyWithBLOBs根据id更新items表中所有字段,包括 大文本类型字段//updateByPrimaryKeyWithBLOBs要求必须转入id
        itemsCustom.setId(id);itemsMapper.updateByPrimaryKeyWithBLOBs(itemsCustom);}}

ItemsServiceImpl.java

    Action

<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:mvc="http://www.springframework.org/schema/mvc"xmlns:context="http://www.springframework.org/schema/context"xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.2.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-3.2.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.2.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.2.xsd http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.2.xsd "><!-- 扫描controller注解,多个包中间使用半角逗号分隔 --><context:component-scan base-package="com.harry.ssm.controller"/><!--注解映射器 --><bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/><!--注解适配器 --><bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/><!-- ViewResolver --><beanclass="org.springframework.web.servlet.view.InternalResourceViewResolver"><property name="viewClass"value="org.springframework.web.servlet.view.JstlView" /><property name="prefix" value="/WEB-INF/jsp/" /><property name="suffix" value=".jsp" /></bean></beans>

springmvc.xml

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd"id="WebApp_ID" version="2.5"><display-name>springmvc</display-name><!-- 加载spring容器 --><context-param><param-name>contextConfigLocation</param-name><param-value>/WEB-INF/classes/spring/applicationContext.xml,/WEB-INF/classes/spring/applicationContext-*.xml</param-value></context-param><listener><listener-class>org.springframework.web.context.ContextLoaderListener</listener-class></listener><!-- 解决post乱码 --><filter><filter-name>CharacterEncodingFilter</filter-name><filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class><init-param><param-name>encoding</param-name><param-value>utf-8</param-value></init-param></filter><filter-mapping><filter-name>CharacterEncodingFilter</filter-name><url-pattern>/*</url-pattern></filter-mapping><!-- springmvc的前端控制器 --><servlet><servlet-name>springmvc</servlet-name><servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class><!-- contextConfigLocation不是必须的, 如果不配置contextConfigLocation, springmvc的配置文件默认在:WEB-INF/servlet的name+"-servlet.xml" --><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>springmvc</servlet-name><url-pattern>*.action</url-pattern></servlet-mapping><welcome-file-list><welcome-file>index.html</welcome-file><welcome-file>index.htm</welcome-file><welcome-file>index.jsp</welcome-file><welcome-file>default.html</welcome-file><welcome-file>default.htm</welcome-file><welcome-file>default.jsp</welcome-file></welcome-file-list>
</web-app>

web.xml

package com.harry.ssm.controller;import java.util.List;import javax.servlet.http.HttpServletRequest;import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.servlet.ModelAndView;import com.harry.ssm.po.ItemsCustom;
import com.harry.ssm.service.ItemsService;@Controller
//为了对url进行分类管理 ,可以在这里定义根路径,最终访问url是根路径+子路径
//比如:商品列表:/items/queryItems.action
@RequestMapping("/items")
public class ItemsController {@Autowiredprivate ItemsService itemsService;// 商品查询@RequestMapping("/queryItems")public ModelAndView queryItems(HttpServletRequest request) throws Exception {//测试forward后request是否可以共享
        System.out.println(request.getParameter("id"));// 调用service查找 数据库,查询商品列表List<ItemsCustom> itemsList = itemsService.findItemsList(null);// 返回ModelAndViewModelAndView modelAndView = new ModelAndView();// 相当 于request的setAttribut,在jsp页面中通过itemsList取数据modelAndView.addObject("itemsList", itemsList);// 指定视图// 下边的路径,如果在视图解析器中配置jsp路径的前缀和jsp路径的后缀,修改为// modelAndView.setViewName("/WEB-INF/jsp/items/itemsList.jsp");// 上边的路径配置可以不在程序中指定jsp路径的前缀和jsp路径的后缀modelAndView.setViewName("items/itemsList");return modelAndView;}//商品信息修改页面显示//@RequestMapping("/editItems")//限制http请求方法,可以post和get
//    @RequestMapping(value="/editItems",method={RequestMethod.POST,RequestMethod.GET})
//    public ModelAndView editItems()throws Exception {
//
//        //调用service根据商品id查询商品信息
//        ItemsCustom itemsCustom = itemsService.findItemsById(1);
//
//        // 返回ModelAndView
//        ModelAndView modelAndView = new ModelAndView();
//
//        //将商品信息放到model
//        modelAndView.addObject("itemsCustom", itemsCustom);
//
//        //商品修改页面
//        modelAndView.setViewName("items/editItems");
//
//        return modelAndView;
//    }
    @RequestMapping(value="/editItems",method={RequestMethod.POST,RequestMethod.GET})//@RequestParam里边指定request传入参数名称和形参进行绑定。//通过required属性指定参数是否必须要传入//通过defaultValue可以设置默认值,如果id参数没有传入,将默认值和形参绑定。public String editItems(Model model,@RequestParam(value="id",required=true) Integer items_id)throws Exception {//调用service根据商品id查询商品信息ItemsCustom itemsCustom = itemsService.findItemsById(items_id);//通过形参中的model将model数据传到页面//相当于modelAndView.addObject方法model.addAttribute("itemsCustom", itemsCustom);return "items/editItems";}//商品信息修改提交@RequestMapping("/editItemsSubmit")public String editItemsSubmit(HttpServletRequest request,Integer id,ItemsCustom itemsCustom)throws Exception {//调用service更新商品信息,页面需要将商品信息传到此方法
        itemsService.updateItems(id, itemsCustom);//重定向到商品查询列表
//        return "redirect:queryItems.action";//页面转发//return "forward:queryItems.action";return "success";}}

ItemsController.java

    View

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%>
<%@ 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" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>查询商品列表</title>
</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><td>商品描述</td><td>操作</td>
</tr>
<c:forEach items="${itemsList }" var="item">
<tr><td>${item.name }</td><td>${item.price }</td><td><fmt:formatDate value="${item.createtime}" pattern="yyyy-MM-dd HH:mm:ss"/></td><td>${item.detail }</td><td><a href="${pageContext.request.contextPath }/items/editItems.action?id=${item.id}">修改</a></td></tr>
</c:forEach></table>
</form>
</body></html>

itemsList.jsp

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%>
<%@ 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" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>修改商品信息</title></head>
<body> <form id="itemForm" action="${pageContext.request.contextPath }/items/editItemsSubmit.action" method="post" >
<input type="hidden" name="id" value="${itemsCustom.id }"/>
修改商品信息:
<table width="100%" border=1>
<tr><td>商品名称</td><td><input type="text" name="name" value="${itemsCustom.name }"/></td>
</tr>
<tr><td>商品价格</td><td><input type="text" name="price" value="${itemsCustom.price }"/></td>
</tr>
<tr><td>商品生产日期</td><td><input type="text" name="createtime" value="<fmt:formatDate value="${itemsCustom.createtime}" pattern="yyyy-MM-dd HH:mm:ss"/>"/></td>
</tr>
<%-- <tr><td>商品图片</td><td><c:if test="${item.pic !=null}"><img src="/pic/${item.pic}" width=100 height=100/><br/></c:if><input type="file"  name="pictureFile"/> </td>
</tr> --%>
<tr><td>商品简介</td><td><textarea rows="3" cols="30" name="detail">${itemsCustom.detail }</textarea></td>
</tr>
<tr>
<td colspan="2" align="center"><input type="submit" value="提交"/>
</td>
</tr>
</table></form>
</body></html>

editItems.jsp

@RequestMapping

  定义于Controller类及其方法上,用于指定访问的url的文件映射路径和访问方法,如

@RequestMapping("/items")
public class ItemsController {

  @RequestMapping("/queryItems",method={RequestMethod.POST,RequestMethod.GET})
  public ModelAndView queryItems(HttpServletRequest request) throws Exception {

那么,指定的路径为http://localhost:8080/appName/items/queryitems.action,有POST和GET两种访问方式

Controller方法返回值

  ModelAndView    对象可添加model数据和指定view

  void         可以利用内部跳转进行响应,或者重定向,或者响应json数据

             request.getRequestDispatcher("页面路径").forward(request, response);

             response.sendRedirect("url")

  String        指定逻辑视图名,重定向,内部跳转

             return "item/editItem";

             return "redirect:queryItem.action";

             return "forward:editItem.action";

参数绑定

   参数绑定可以是客户端数据绑定,也可以是服务端参数绑定;

  服务端参数绑定就是在request<--controller这个过程之间发生的事,request到达controller控制权就在我们的手上,我们可以运行自己的逻辑代码,然后获取参数,然后再将其绑定在request域中,等着给DispatchServlet渲染.

  支持的类型:HttpServletRequest,HttpServletResponse,HttpSession,Model/ModelMap

Items item = itemService.findItemById(id);

model.addAttribute("item", item);

modelMap是model接口实现类,实质上都是实例化ModelMap来存储key/value

页面通过${item.xxx}来取值.

  客户端参数绑定就是request-->controller,将客户填入的数据进行自动获取至方法上

  支持的类型:基础类型(Bollean,Integer,String,Float/Double),POJO,自定义类型转换器,集合类

public String editItem(Model model,Integer id,Boolean status) throws Exception
http://localhost:8080/springmvc_mybatis/item/editItem.action?id=2&status=false

参数id和status将自动获取

@RequestParam 常用来处理基础类型

public String editItem(@RequestParam(value="item_id",required=true) String id

  @RequestParam(value="item_status",defaultValue="true") Boolean status) {}

页面定义

<input type="text" name="name"/>

<input type="text" name="price"/>

Controller方法定义,获取items匹配的属性名name,price

@RequestMapping("/editItemSubmit")

public String editItemSubmit(Items items)throws Exception{

System.out.println(items);

自定义类型转换器

public class CustomDateConverter implements Converter<String, Date> {@Overridepublic Date convert(String source) {try {SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");return simpleDateFormat.parse(source);} catch (Exception e) {e.printStackTrace();}return null;}}

CustomDateConverter.java

<mvc:annotation-driven conversion-service="conversionService">
</mvc:annotation-driven>
<!-- conversionService --><bean id="conversionService"class="org.springframework.format.support.FormattingConversionServiceFactoryBean"><!-- 转换器 --><property name="converters"><list><bean class="com.harry.controller.converter.CustomDateConverter"/></list></property></bean>

springmvc.xml

  

  集合类

<input type="checkbox" name="item_id" value="001"/>
<input type="checkbox" name="item_id" value="002"/>
<input type="checkbox" name="item_id" value="003"/>-------------------------------------------------------------------------
public String deleteitem(String[] item_id)throws Exception{System.out.println(item_id);
}

StringArray

<c:forEach items="${itemsList }" var="item" varStatus="s">
<tr><td><input type="text" name="itemsList[${s.index }].name" value="${item.name }"/></td><td><input type="text" name="itemsList[${s.index }].price" value="${item.price }"/></td>..........
</tr>
</c:forEach>
-----------------------------------------------------------------------------------------------------
public class QueryVo {
private List<Items> itemList;//商品列表//get/set方法..
}-----------------------------------------------------------------------------------------------------
public String useraddsubmit(Model model,QueryVo queryVo)throws Exception{
System.out.println(queryVo.getItemList());
}

List

<tr>
<td>学生信息:</td>
<td>
姓名:<inputtype="text"name="itemInfo['name']"/>
年龄:<inputtype="text"name="itemInfo['price']"/>
.. .. ..
</td>
</tr>
----------------------------------------------------------------------------------------
public class QueryVo {
private Map<String, Object> itemInfo = new HashMap<String, Object>();//get/set方法..
}
-----------------------------------------------------------------------------------------
public String useraddsubmit(Model model,QueryVo queryVo)throws Exception{
System.out.println(queryVo.getStudentinfo());
}

Map

  

自定义类与包装类

  我们在工作中一般使用逆向工程生成工具来进行代码开发,也就是ORM里面的实体部分已经是自动生成的,这部分的代码不能动,因为很容易造成代码混乱,但是我们又经常需要去修改里面的部分属性;

  我们经常还碰到另外一种情况,就是客户端请求的可能是包含几个类的视图(如多值查询),那么针对这种情况,我们可以使用一个包装类来作为视图类,然后返回视图类来进行响应,而我们上面也是这么做的.

  鉴于上述原因,我们会使用自定义类来对其进行封装,使用视图类来进行传输,这样,我们就可以在不碰那些代码的前提下任意进行代码修改.

  

  

  

  

编码问题

  相信你在开发时一定碰到过乱码问题,SpringMVC也提供了一个统一的编码过滤器,但是只适用于POST编码过滤,这是因为开发中上传表单数据都规定使用POST,所以SpringMVC也就只提供这一种过滤器.

<filter>
<filter-name>CharacterEncodingFilter</filter-name>
<filter-class>org.springframework.web.filter.CharacterEncodingFilter</filter-class>
<init-param>
<param-name>encoding</param-name>
<param-value>utf-8</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>CharacterEncodingFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

web.xml

  还有GET的乱码解决方案有两种,

  1.修改tomcat配置文件添加编码与工程编码一致

  <Connector URIEncoding="utf-8" connectionTimeout="20000" port="8080" protocol="HTTP/1.1" redirectPort="8443"/>

  2.对参数进行重新编码  

  String userName = new String(request.getParamter("userName").getBytes("ISO8859-1"),"utf-8");

Validation

  数据校验一般都会在前端进行,但是有时安全性要求高点的环境下,也会在服务端进行数据校验.

  服务端校验

  Controller层  检验页面请求的合法性,在服务端控制层conroller校验,不区分客户端类型(浏览器、手机客户端、远程调用)

  Service层   不区分客户端类型(浏览器、手机客户端、远程调用)

  DAO层     一般不进行校验

  

  SpringMVC使用校验框架Validation.

  校验思路:
  页面提交请求的参数,请求到controller方法中,使用validation进行校验.如果校验出错,将错误信息展示到页面.

  1.导入Bean-Validation框架jar包,Hibernate-validator

  2.配置校验器

<!-- 校验器 --><bean id="validator"class="org.springframework.validation.beanvalidation.LocalValidatorFactoryBean"><!-- hibernate校验器--><property name="providerClass" value="org.hibernate.validator.HibernateValidator" /><!-- 指定校验使用的资源文件,在文件中配置校验错误信息,如果不指定则默认使用classpath下的ValidationMessages.properties --><property name="validationMessageSource" ref="messageSource" /></bean>
<!-- 校验错误信息配置文件 --><bean id="messageSource"class="org.springframework.context.support.ReloadableResourceBundleMessageSource"><!-- 资源文件名--><property name="basenames">   <list>    <value>classpath:CustomValidationMessages</value> </list>   </property><!-- 资源文件编码格式 --><property name="fileEncodings" value="utf-8" /><!-- 对资源文件内容缓存时间,单位秒 --><property name="cacheSeconds" value="120" /></bean><!--注解映射器 --><!-- <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/> --><!--注解适配器 --><!-- <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/> --><!-- 使用 mvc:annotation-driven代替上边注解映射器和注解适配器配置mvc:annotation-driven默认加载很多的参数绑定方法,比如json转换解析器就默认加载了,如果使用mvc:annotation-driven不用配置上边的RequestMappingHandlerMapping和RequestMappingHandlerAdapter实际开发时使用mvc:annotation-driven--><mvc:annotation-driven conversion-service="conversionService"validator="validator"></mvc:annotation-driven>

springmvc.xml

  3.在POJO中添加校验规则.

  

  4.配置校验时产生的message

  

  5.捕获校验错误信息

    

  在需要校验的pojo前边添加@Validated,在需要校验的pojo后边添加BindingResult bindingResult接收校验出错信息
  注意:@Validated和BindingResult bindingResult是配对出现,并且形参顺序是固定的(一前一后).

  6.controller传递错误信息

  

  7.页面显示错误信息

  

  8.校验分组

  在pojo中定义校验规则,而pojo是被多个 controller所共用,当不同的controller方法对同一个pojo进行校验,但是每个controller方法需要不同的校验.可以指定不同的分组来进行校验.

  

  9.在校验中添加分组

  

  10.在Controller指定分组的校验规则

  

数据回显

  数据回显指提交数据后出现错误,将刚才提交的数据回显到刚才的提交页面,比如填一张表,填完后一项错误后重新回到提交前的状态.

  1.SpringMVC默认对POJO进行回显.

  pojo数据传入controller方法后,springmvc自动将pojo数据放到request域,key等于pojo类型(首字母小写)

  

  

  2.@ModelAttribute

  @ModelAttribute可以指定POJO对应的key值,页面获取可以直接使用key值进行获取,还可以将返回值存入request域

  

  

  3.直接使用model.addAttribute

  

异常处理器

  异常处理思路很简单,就是DAO,Service,Controller出现异常直接向上抛出,然后在DispatchServlet那一层设置一个异常处理器就处理所有异常.

  

  1.创建自定义Exception  

package com.harry.exception;@SuppressWarnings("serial")
public class ExceptionCustom extends Exception {//异常信息public String message;public ExceptionCustom(String message){super(message);this.message = message;}public String getMessage() {return message;}public void setMessage(String message) {this.message = message;}
}

ExceptionCustom.java

  2.定义全局异常处理器  

  全局异常处理器处理思路:
    解析出异常类型;
    如果该异常类型是系统自定义的异常直接取出异常信息,在错误页面展示;
    如果该异常类型不是系统自定义的异常,构造一个自定义的异常类(信息为"未知错误");

  实现SpringMVC提供的HandlerExceptionResolver接口

package com.harry.exception;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import org.springframework.web.servlet.HandlerExceptionResolver;
import org.springframework.web.servlet.ModelAndView;public class ExceptionCustomResolver implements HandlerExceptionResolver {/*** (非 Javadoc)* <p>Title: resolveException</p>* <p>Description: </p>* @param request* @param response* @param handler* @param ex 系统 抛出的异常* @return* @see org.springframework.web.servlet.HandlerExceptionResolver#resolveException(javax.servlet.http.HttpServletRequest, javax.servlet.http.HttpServletResponse, java.lang.Object, java.lang.Exception)*/@Overridepublic ModelAndView resolveException(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex) {//handler就是处理器适配器要执行Handler对象(只有method)//        解析出异常类型
//        如果该 异常类型是系统 自定义的异常,直接取出异常信息,在错误页面展示
//        String message = null;
//        if(ex instanceof ExceptionCustom){
//            message = ((ExceptionCustom)ex).getMessage();
//        }else{            如果该 异常类型不是系统 自定义的异常,构造一个自定义的异常类型(信息为“未知错误”)
//            message="未知错误";
//        }//上边代码变为ExceptionCustom ExceptionCustom = null;if(ex instanceof ExceptionCustom){ExceptionCustom = (ExceptionCustom)ex;}else{ExceptionCustom = new ExceptionCustom("未知错误");}//错误信息String message = ExceptionCustom.getMessage();ModelAndView modelAndView = new ModelAndView();//将错误信息传到页面modelAndView.addObject("message", message);//指向错误页面modelAndView.setViewName("error");return modelAndView;}}

ExceptionCustomResolver.java

  错误页面

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>错误提示</title>
</head>
<body>
${message }
</body>
</html>

error.jsp

  springmvc.xml配置全局异常处理器

<!-- 全局异常处理器只要实现HandlerExceptionResolver接口就是全局异常处理器--><bean class="cn.itcast.ssm.exception.CustomExceptionResolver"></bean>

springmvc.xml

上传图片

  1.当提交的表单数据混合几种数据格式,比如字符串和图片文件,必须在表单设置提交的压缩类型为multipart/form-data.

  

  2.要解析这种压缩格式的数据需要在springmvc配置文件中添加multipart类型解析器.

<!-- 文件上传 --><bean id="multipartResolver"class="org.springframework.web.multipart.commons.CommonsMultipartResolver"><!-- 设置上传文件的最大尺寸为5MB --><property name="maxUploadSize"><value>5242880</value></property></bean>

springmvc.xml

  3.加上springmvc图片组件jar包

  

  4.创建图片虚拟目录存储图片

  

  或者在tomcat/conf/server.xml下

  

注意:在图片虚拟目录 中,一定将图片目录分级创建(提高i/o性能),一般我们采用按日期(年-月-日)进行分级创建.

  5.上传页面部分代码  

<tr><td>商品图片</td><td><c:if test="${item.pic !=null}"><img src="/pic/${item.pic}" width=100 height=100/><br/></c:if><input type="file"  name="pictureFile"/> </td>
</tr>

editItems.jsp

  6.添加controller映射model,保存图片路径名称

  

  

Json数据交互

  Json数据交互的应用场景广泛,基本上每个网站都会有这样的数据交互,特点是数据交互量小,轻便,可异步刷新.

  1.json数据交互思路

  

  客户端请求json串(或者键值对) <--------> 服务端json对象

  2.springmvc采用jackson包来转换json串与json对象,所以要添加jar包

  

  3.配置json转换器

<!--注解适配器 --><bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"><property name="messageConverters"><list><bean class="org.springframework.http.converter.json.MappingJacksonHttpMessageConverter"></bean></list></property></bean><!--如果使用<mvc:annotation-driven /> 则不用定义上边的内容 -->

springmvc.xml

  4.测试请求json串和key/value,输出json串

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>json交互测试</title>
<script type="text/javascript" src="${pageContext.request.contextPath }/js/jquery-1.4.4.min.js"></script>
<script type="text/javascript">
//请求json,输出是json
function requestJson(){$.ajax({type:'post',url:'${pageContext.request.contextPath }/requestJson.action',contentType:'application/json;charset=utf-8',//数据格式是json串,商品信息data:'{"name":"手机","price":999}',success:function(data){//返回json结果
            alert(data);}});}
//请求key/value,输出是json
function requerstKeyValue(){$.ajax({type:'post',url:'${pageContext.request.contextPath }/requerstKeyValue.action',//请求是key/value这里不需要指定contentType,因为默认就 是key/value类型//contentType:'application/json;charset=utf-8',//数据格式是json串,商品信息data:'name=手机&price=999',success:function(data){//返回json结果
            alert(data.name);}});}
</script>
</head>
<body>
<input type="button" οnclick="requestJson()" value="请求json,输出是json"/>
<input type="button" οnclick="requerstKeyValue()" value="请求key/value,输出是json"/>
</body>
</html>

jsonTest.jsp

package com.harry.controller;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;import com.harry.entity.ItemsCustom;@Controller
public class JsonController {//请求json串(商品信息),输出json(商品信息)//@RequestBody将请求的商品信息的json串转成itemsCustom对象//@ResponseBody将itemsCustom转成json输出@RequestMapping("/requestJson")public @ResponseBody ItemsCustom requestJson(@RequestBody ItemsCustom itemsCustom){//@ResponseBody将itemsCustom转成json输出return itemsCustom;}//请求key/value,输出json@RequestMapping("/responseJson")public @ResponseBody ItemsCustom responseJson(ItemsCustom itemsCustom){//@ResponseBody将itemsCustom转成json输出return itemsCustom;}}

JsonController.java

  

RESTful支持

  RESTful是一种开发理念,并非指具体的技术,理念的阐述具体如下:

  1.对url进行规范,写RESTful格式的url

  非REST的url:http://...../queryItems.action?id=001&type=T01

  REST的url风格:http://..../items/001

特点:url简洁,将参数通过url传到服务端

  2.http的方法规范

  不管是删除,添加,更新,使用url是一致的,如果进行删除,需要设置http的方法为delete,同理添加.

  后台controller方法:判断http方法,如果是delete执行删除,如果是post执行添加。

  3.对http的contentType规范

  请求时指定contentType,要json数据,设置成json格式的type.

  示例

  1.controller定义映射使用RESTful风格的url,将查询信息的id传入controller.

  

  @RequestMapping(value="/ itemsView/{id}"):{×××}占位符,请求的URL可以是“/viewItems/1”或“/viewItems/2”,通过在方法中使用@PathVariable获取{×××}中的×××变量.

   @PathVariable用于将请求URL中的模板变量映射到功能处理方法的参数上. 如果RequestMapping中表示为"/ itemsView /{id}",id和形参名称一致,@PathVariable不用指定名称.

  2.配置前端控制器

  

  "*.action"的解析在RESTful风格下会导致静态资源无法访问,这里替换成"/";

  3.springmvc.xml中配置静态资源映射

  

拦截器

拦截器能设置请求前,响应后进行拦截执行特定的代码,SpringMVC定义拦截器需要实现HandlerInterceptor接口.

接口中提供三个方法分别在handler执行前,后还有ModelAndView返回前执行

package com.harry.interceptor;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;public class HandlerInterceptor1 implements HandlerInterceptor {//进入 Handler方法之前执行//用于身份认证、身份授权//比如身份认证,如果认证通过表示当前用户没有登陆,需要此方法拦截不再向下执行
    @Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {System.out.println("HandlerInterceptor1...preHandle");//return false表示拦截,不向下执行//return true表示放行return true;}//进入Handler方法之后,返回modelAndView之前执行//应用场景从modelAndView出发:将公用的模型数据(比如菜单导航)在这里传到视图,也可以在这里统一指定视图
    @Overridepublic void postHandle(HttpServletRequest request,HttpServletResponse response, Object handler,ModelAndView modelAndView) throws Exception {System.out.println("HandlerInterceptor1...postHandle");}//执行Handler完成执行此方法//应用场景:统一异常处理,统一日志处理
    @Overridepublic void afterCompletion(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex)throws Exception {System.out.println("HandlerInterceptor1...afterCompletion");}}

HanlerInterceptor1.java

  针对HandlerMapping的拦截器(不推荐使用)

  完成拦截器编写后需要在HandlerMapping处进行定义,毕竟拦截器是映射阶段进行工作的.

package com.harry.interceptor;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;public class HandlerInterceptor1 implements HandlerInterceptor {//进入 Handler方法之前执行//用于身份认证、身份授权//比如身份认证,如果认证通过表示当前用户没有登陆,需要此方法拦截不再向下执行
    @Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {System.out.println("HandlerInterceptor1...preHandle");//return false表示拦截,不向下执行//return true表示放行return true;}//进入Handler方法之后,返回modelAndView之前执行//应用场景从modelAndView出发:将公用的模型数据(比如菜单导航)在这里传到视图,也可以在这里统一指定视图
    @Overridepublic void postHandle(HttpServletRequest request,HttpServletResponse response, Object handler,ModelAndView modelAndView) throws Exception {System.out.println("HandlerInterceptor1...postHandle");}//执行Handler完成执行此方法//应用场景:统一异常处理,统一日志处理
    @Overridepublic void afterCompletion(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex)throws Exception {System.out.println("HandlerInterceptor1...afterCompletion");}}

springmvc.xml

  全局性拦截器

<!--拦截器 -->
<mvc:interceptors><!--多个拦截器,顺序执行 --><!-- 登陆认证拦截器 --><mvc:interceptor><mvc:mapping path="/**"/><bean class="com.harry.interceptor.LoginInterceptor"></bean></mvc:interceptor><mvc:interceptor><!-- /**表示所有url包括子url路径 --><mvc:mapping path="/**"/><bean class="com.harry.interceptor.HandlerInterceptor1"></bean></mvc:interceptor><mvc:interceptor><mvc:mapping path="/**"/><bean class="com.harry.interceptor.HandlerInterceptor2"></bean></mvc:interceptor>
</mvc:interceptors>

springmvc.xml

  拦截器的应用 

  比如:统一日志处理拦截器,需要该 拦截器preHandle一定要放行,且将它放在拦截器链接中第一个位置.

  比如:登陆认证拦截器,放在拦截器链接中第一个位置.权限校验拦截器,放在登陆认证拦截器之后.(因为登陆通过后才校验权限)

  拦截器实现登陆认证

1.处理逻辑

  用户请求页面

  拦截器进行拦截

    如果页面无需访问权限,放行

    页面需要访问权限

      session存在校验数据,通过

      session无校验数据,转入登陆页面

 2.登陆用Controller方法和请求页面  

package com.harry.controller;import javax.servlet.http.HttpSession;import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;@Controller
public class LoginController {// 登陆@RequestMapping("/login")public String login(HttpSession session, String username, String password)throws Exception {// 调用service进行用户身份验证// ...// 在session中保存用户身份信息session.setAttribute("username", username);// 重定向到商品列表页面return "redirect:/items/queryItems.action";}// 退出@RequestMapping("/logout")public String logout(HttpSession session) throws Exception {// 清除session
        session.invalidate();// 重定向到商品列表页面return "redirect:/items/queryItems.action";}}

LoginController.java

<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<title>系统登陆</title>
</head>
<body>
<form action="${pageContext.request.contextPath }/login.action" method="post">
用户账号:<input type="text" name="username" /><br/>
用户密码 :<input type="password" name="password" /><br/>
<input type="submit" value="登陆"/>
</form>
</body>
</html>

login.jsp

   3.登陆拦截器和设置

package com.harry.interceptor;import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession;import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;public class LoginInterceptor implements HandlerInterceptor {//进入 Handler方法之前执行//用于身份认证、身份授权//比如身份认证,如果认证通过表示当前用户没有登陆,需要此方法拦截不再向下执行
    @Overridepublic boolean preHandle(HttpServletRequest request,HttpServletResponse response, Object handler) throws Exception {//获取请求的urlString url = request.getRequestURI();//判断url是否是公开 地址(实际使用时将公开 地址配置配置文件中)//这里公开地址是登陆提交的地址if(url.indexOf("login.action")>=0){//如果进行登陆提交,放行return true;}//判断sessionHttpSession session  = request.getSession();//从session中取出用户身份信息String username = (String) session.getAttribute("username");if(username != null){//身份存在,放行return true;}//执行这里表示用户身份需要认证,跳转登陆页面request.getRequestDispatcher("/WEB-INF/jsp/login.jsp").forward(request, response);//return false表示拦截,不向下执行//return true表示放行return false;}//进入Handler方法之后,返回modelAndView之前执行//应用场景从modelAndView出发:将公用的模型数据(比如菜单导航)在这里传到视图,也可以在这里统一指定视图
    @Overridepublic void postHandle(HttpServletRequest request,HttpServletResponse response, Object handler,ModelAndView modelAndView) throws Exception {System.out.println("HandlerInterceptor1...postHandle");}//执行Handler完成执行此方法//应用场景:统一异常处理,统一日志处理
    @Overridepublic void afterCompletion(HttpServletRequest request,HttpServletResponse response, Object handler, Exception ex)throws Exception {System.out.println("HandlerInterceptor1...afterCompletion");}}

LoginInterceptor.java

        <!--拦截器 -->
<mvc:interceptors><!--多个拦截器,顺序执行 --><!-- 登陆认证拦截器 --><mvc:interceptor><mvc:mapping path="/**"/><bean class="com.harry.interceptor.LoginInterceptor"></bean></mvc:interceptor>
</mvc:interceptors>

springmvc.xml

转载于:https://www.cnblogs.com/CARPE-DIEM-wu/p/7643446.html

框架应用 : Spring MVC - 开发详述相关推荐

  1. MyBatis+Spring MVC开发指南(一)

    前言 MyBatis+Spring MVC这套组合,在实际互联网项目中非常流行,博主工作中也涉及过,打算由浅入深.系统的写出来!这个系列将会涵盖MyBatis开发详解.Spring MVC开发详解,以 ...

  2. 使用Spring MVC开发Restful Web服务

    REST简介 摘自Wikipedia: REST风格的体系结构由客户端和服务器组成. 客户端向服务器发起请求: 服务器处理请求并返回适当的响应. 请求和响应围绕资源表示的传递而构建. 资源本质上可以是 ...

  3. Spring MVC开发步骤以及执行流程

    Spring MVC开发步骤以及执行流程 开发步骤 1.在web.xml中定义前端控制器DispatcherServlet来拦截用户请求. 2.如果要以post方式提交请求,则定义包含表单数据的jsp ...

  4. spring boot之Spring Mvc开发和常用注解说明

    spring boot之Spring Mvc开发和常用注解说明 Spring boot主要用于开发微服务接口.所以采用的控制层注解主要是restful形式. 创建微服务: 与上一篇文章一致. 依赖多一 ...

  5. Spring MVC开发–快速教程

    这是我们的JCG合作伙伴之一,来自Manoj的有关使用Spring开发Web应用程序的简短教程, 网址为" The Khangaonkar Report ". (注意:对原始帖子进 ...

  6. SSM框架之Spring MVC(四)异常处理和拦截器实现

    一.SpringMVC中的异常处理 1.1异常处理的思路 系统中异常包括两类:预期异常和运行时异常 RuntimeException,前者通过捕获异常从而获取异常信息,后者主要通过规范代码开发.测试通 ...

  7. SSM框架之Spring MVC(一)

    一.Spring MVC简单介绍: 1.1 Spring MVC概述 SpringMVC是一种基于Java的实现MVC(Model View Controller)设计模型的请求驱动类型的轻量级Web ...

  8. Spring MVC-使用Spring Tool Suite IDE搭建Spring MVC开发环境

    Spring MVC 概述 新建Spring MVC Project 分析IDE建立的工程 Maven dependencies configuration Spring MVC configurat ...

  9. Spring MVC开发RESTful风格的URI

    一.写在前面 RESTful结构可参考博文:https://blog.csdn.net/codejas/article/details/79799386 我们知道在HTTP 协议中,有四种操作方式的动 ...

最新文章

  1. SL2_RedSkin
  2. leetcode-237-删除链表中的节点
  3. Uva 11600 期望DP
  4. Machine Learning on Spark—— 统计基础(一)
  5. mysql ==null_mysql = null 问题
  6. 单行文字、多行文字溢出时省略号表示的多种解决方式;调整字符间距;段落首字母大写缩进效果;
  7. 安装配置ASMlib驱动
  8. c#winform演练 ktv项目 MediaPlay控件的暂停播放与停止
  9. 为啥国内程序员写的代码也用英文注释?
  10. 一层循环时间复杂度_数据结构与算法系列——时间、空间复杂度
  11. 使用Hadoop搭建现代电信企业架构
  12. iOS官方demo下载网站
  13. python是一种编译型程序设计语言_Python语言是一种编译型高级程序设计语言。
  14. windows使用DD刻录工具刻录U盘
  15. 阿里天池大数据竞赛——口碑商家客流量预测 A
  16. 杭州电子科技大学java刷题_杭州电子科技大学的OJ
  17. Excel制作图表(二)--- 燃尽图
  18. “探月计划”来袭,美国米德天文望远镜助孩子观月赏月
  19. Linux下Rootkit介绍
  20. linux路由配置秘籍,最新最全秘籍:如何提升路由器信号强度?

热门文章

  1. Oracle 同义词(synonym) 简介
  2. java8生成jsp页面内容组装到jsp中_JAVA WEB快速入门之从编写一个JSP WEB网站了解JSP WEB网站的基本结构、调试、部署...
  3. Spring Cloud Gateway 突发高危漏洞,下一代云原生网关恰逢其时?
  4. 课时 11:可观测性:你的应用健康吗?(莫源)
  5. 应用系统瓶颈排查和分析的思考-Arthas 实战
  6. 阿里百万级规模开源容器 PouchContainer GA 版本发布,邀您参与上海 Meetup 共话容器未来
  7. qt5 linux 控制台 乱码,qt5.12 解决显示中文乱码问题
  8. freertos 定时器 不启动_FreeRTOS 从入门到精通8--软件定时器应用
  9. git clone github_Git下载速度慢的解决方案汇总
  10. python如何使用图片做背景_用Python批量给照片换底色,基于opencv模块