springmvc(一) springmvc框架原理分析和简单入门程序
springmvc这个框架真的非常简单,感觉比struts2还更简单,好好沉淀下来学习~
--WZY
一、什么是springmvc?
我们知道三层架构的思想,并且如果你知道ssh的话,就会更加透彻的理解这个思想,struts2在web层,spring在中间控制,hibernate在dao层与数据库打交道,而前面刚写的mybatis跟hibernate一样,与数据库打交道在dao层的另一个框架,而今天所要讲解的springmvc是在web层的另一个框架。
springmvc全名是spring web mvc,springmvc是spring的一个模块,并且看名字即可知道,springmvc是一个基于mvc设计模式的前端web框架。
mvc:m(model模型)、v(view视图)、c(control控制)
mvc的运用概念图
二、springmvc的入门程序
通过这个来快速了解springmvc大概的开发流程,其实通过上面的mvc分析图,差不多就知道了如何开发了。重点就是三步。
1、在web.xml中配置一个serlvet,用来控制,
2、编写一个handler(controller)类,用来做业务处理。
3、编写jsp或者别的视图,用来展示数据
思路已经有了,那么就开始编写把。
问题描述:使用springmvc来完成前端请求的处理
2.1、创建web工程
2.2、添加jar包
2.3、编程步骤
前面三步只是通过mvc图的分析出最关键的三步,其中实现的时候步骤应该更多,比如spring的配置文件,但关键的重点还是那三个。
1、创建po类
2、配置前端控制器,DispatcherServlet
3、创建springmvc的配置文件
4、开发handler(controller)
5、在springmvc的配置文件中(取名为springmvc.xml)配置handler
6、开发jsp或者别的视图
8、部署测试
2.4、创建po类
2.5、配置前端控制器
1 <!-- springmvc 的前端控制器 --> 2 <servlet> 3 <servlet-name>springmvc</servlet-name> 4 <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> 5 <!-- 指定springmvc的配置文件的地址 --> 6 <init-param> 7 <param-name>contextConfigLocation</param-name> 8 <param-value>classpath:springmvc.xml</param-value> 9 </init-param> 10 </servlet> 11 <servlet-mapping> 12 <servlet-name>springmvc</servlet-name> 13 <!-- 这里有三种配置url-pattern方案 14 1、*.do:后缀为.do的请求才能够访问到该servlet[用这个] 15 2、/ :所有请求都能够访问到该servlet(除jsp),包括静态请求(处理会有问题,不用) 16 3、/* :有问题,因为访问jsp也会到该servlet,而访问jsp时,我们不需要这样,也不用 17 --> 18 <url-pattern>*.do</url-pattern> 19 </servlet-mapping>
2.6、创建springmvc的配置文件
在config目录下,创建springmvc.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 "> </beans>
2.7、开发handler(controller)类,也就是处理业务逻辑的类,
2.8、在springmvc.xml中配置handler类,也就是spring帮我们创建该类的实例,所以需要配置。
注意:2.7和2.8一起讲解,因为开发handler类讲解三种方式,所以配置也连在一起讲解,以免分开来,看不清楚
Springmvc开发handler有多种方式,我们只讲解三种:实现HttpRequestHandler接口、实现Controller接口、使用注解开发(掌握)
实现HttpRequestHandler接口
1 package com.wuhao.springmvc.controller; 2 3 import java.io.IOException; 4 import java.util.ArrayList; 5 import java.util.List; 6 7 import javax.servlet.ServletException; 8 import javax.servlet.http.HttpServletRequest; 9 import javax.servlet.http.HttpServletResponse; 10 11 import org.springframework.web.HttpRequestHandler; 12 13 import com.wuhao.springmvc.domain.Items; 14 15 public class ItemController implements HttpRequestHandler { 16 17 @Override 18 public void handleRequest(HttpServletRequest request, HttpServletResponse response) 19 throws ServletException, IOException { 20 //获取商品列表(用静态数据模拟) 21 List<Items> itemsList = new ArrayList<Items>(); 22 23 Items items_1 = new Items(); 24 items_1.setName("联想笔记本 HttpRequestHandler"); 25 items_1.setPrice(6000f); 26 items_1.setDetail("ThinkPad T430 联想笔记本电脑!"); 27 28 Items items_2 = new Items(); 29 items_2.setName("苹果手机"); 30 items_2.setPrice(5000f); 31 items_2.setDetail("iphone6苹果手机!"); 32 33 itemsList.add(items_1); 34 itemsList.add(items_2); 35 36 //把商品数据放到request域中 37 request.setAttribute("itemsList", itemsList); 38 //指定视图 39 request.getRequestDispatcher("/WEB-INF/jsp/items/itemsList.jsp").forward(request, response); 40 41 } 42 43 }
springmvc.xml中配置该处理器
通过localhost:8080/项目名/queryItems01.do 就能够访问到DispatcherSerlvet,该servlet就会帮我们找到你对应的处理器(依据就是通过下面的这行配置,queryItems01对应了一个处理器的class,也就能够找到)
1 <!-- 配置实现HttpRequestHander接口的处理器 --> 2 <bean name="/queryItems01.do" class="com.wuhao.springmvc.controller.ItemController"></bean>
实现Controller接口
1 package com.wuhao.springmvc.controller; 2 3 import java.util.ArrayList; 4 import java.util.List; 5 6 import javax.servlet.http.HttpServletRequest; 7 import javax.servlet.http.HttpServletResponse; 8 9 import org.springframework.web.servlet.ModelAndView; 10 import org.springframework.web.servlet.mvc.Controller; 11 12 import com.wuhao.springmvc.domain.Items; 13 14 public class ItemController02 implements Controller { 15 16 @Override 17 public ModelAndView handleRequest(HttpServletRequest request, 18 HttpServletResponse response) throws Exception { 19 //获取商品列表(用静态数据模拟) 20 List<Items> itemsList = new ArrayList<Items>(); 21 22 Items items_1 = new Items(); 23 items_1.setName("联想笔记本 Controller"); 24 items_1.setPrice(6000f); 25 items_1.setDetail("ThinkPad T430 联想笔记本电脑!"); 26 27 Items items_2 = new Items(); 28 items_2.setName("苹果手机"); 29 items_2.setPrice(5000f); 30 items_2.setDetail("iphone6苹果手机!"); 31 32 itemsList.add(items_1); 33 itemsList.add(items_2); 34 35 //实现Controller接口的话,就必须使用MoldeAndView对象来将数据装载到对应的jsp视图上,然后返回该对象即可 36 //所以需要两步,将数据给该对象,将指定的视图在交给该对象,最后返回该对象即可。 37 ModelAndView mv = new ModelAndView(); 38 //类似于request.setAttribute("itemsList", itemsList); 39 mv.addObject("itemsList", itemsList); 40 41 //指定视图 42 mv.setViewName("/WEB-INF/jsp/items/itemsList.jsp"); 43 44 return mv; 45 46 } 47 48 }
配置该handler类
<!-- 配置实现Controller接口的处理器 -->
<bean name="/queryItems02.do" class="com.wuhao.springmvc.controller.ItemController02"></bean>
使用注解开发
注解的配置,就是配置一个扫描器,扫描使用了注解的地方
<!-- 使用注解的handle,则需要配置组件扫描器,加载handler base-package:指定要扫描的包 --> <context:component-scan base-package="com.wuhao.springmvc.controller" ></context:component-scan>
2.9、开发jsp
在WEB-INF/jsp/items/下创建jsp: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 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 }/editItems.do?id=${item.id}">修改</a></td> </tr> </c:forEach> </table> </form> </body> </html>
2.10、部署测试
测试上面三种采用不同的方式编写的处理类。能够成功访问即成功
三、springmvc框架原理图分析
前面了解了springmvc的mvc设计模式的运用并且还编写了一个简单的实例,关键点就几个,配置DispatcherServlet,编写处理类以及配置,jsp,就mvc的三个关键点,但是这也是粗略的使用一下springmvc,并不知道其中运行的原理,比如
springmvc是如何找到处理器的?
springmvc如何执行处理器的?
springmvc如何查找到视图对象的?
看图即可
1、发起请求到前端控制器(DispatcherServlet),该控制器中就会过滤出你哪些请求可以访问该servlet哪些不可以,就是url-pattern的作用,并且会加载springmvc.xml配置文件
2、前端控制器会找到HandlerMapping(处理器映射器),通过HandlerMapping完成url到controller映射的组件,通俗点讲,就是将在springmvc.xml中配置的或者注解的url与对应的处理类找到并进行存储,实际上是用一个map集合来保存这种映射关系,map<url,handler>; 这样,就将所有的这种映射关系都记录保存了下来
3、通过HandlerMapping有了这些映射关系,并且找到了url对应的处理器,HandlerMapping就会将其处理器(图中红色标明的handler)返回,在其返回之前,在加上很多的拦截器,其作用后面
进行讲解,这里知道在返回的处理器前会有很多的拦截器即可。
4、DispatcherServlet拿到了handler之后,找到HandlerAdapter(处理器适配器),通过它来访问处理器,并且执行处理器。
这里会有人会有疑惑,为什么需要处理器适配器,我们都获得了处理类了,直接调用不就行了吗?
不行,因为我们只知道处理类在哪里,并不知道执行处理类中的哪个方法,其实也就是不知道处理类是通过哪种方式创建出来的,实现HttpRequestHandler?还是注解方式,或者是 其他方式,我们不知道,所以需要HandlerAdapter来帮我们确认调用哪个方法。
5、执行处理器
6、处理器会返回一个ModelAndView对象给HandlerAdapter
7、通过HandlerAdapter将ModelAndView对象返回给前端控制器(DispatcherServlet)
8、前端控制器请求视图解析器(ViewResolver)去进行视图解析,根据逻辑视图名解析成真正的视图(jsp),其实就是将ModelAndView对象中存放视图的名称进行查找,找到对应的页面形成视图对象
9、返回视图对象到前端控制器。
10、视图渲染,就是将ModelAndView对象中的数据放到request域中,用来让页面加载数据的。
11、通过第8步,通过名称找到了对应的页面,通过第10步,request域中有了所需要的数据,那么就能够进行视图渲染了。最后将其返回即可。
通过上面的图和分析过程,就能够完美解答上面的三个问题了。理解了图,那么springmvc就会用了。很简单把,跟struts2差不多,记住原理图即可。
四、组件分析(默认组件和手动配置组件)
通过图可以看到
前端控制器:对其他组件进行解耦,这样就增加了组件的可扩展性 无需开发直接配置
处理器映射器:无需开发,直接用,作用见上面
处理器适配器:无需开发,
处理器:需要开发,方式很多
视图解析器:无需开发
视图:需要开发
就这么点东西,真正需要写的就两个(处理器+视图)和一个配置(前端控制器),就是mvc中的三个重点,在第二小节中就是这样编写的,第三小结就是解释其中的原理。
处理器映射器、处理器适配器、视图解析器这三个是默认配置的,在下面位置中可以查看
DispatchServlet.properties
上面是使用默认的,如果没有配置这几个组件,那么就使用默认的,我们也可以手动指定。
4.1、非注解的处理器映射器和处理器适配器 [看看即可]
BeanNameUrlHandlerMapping:映射器
在springmvc配置文件中,配置BeanNameUrlHandlerMapping
他的作用是找到在springmvc.xml中配置的url和处理器的bean
HttpRequestHandlerAdapter:适配执行实现了HttpRequestHandler接口的处理类的方法
在springmvc配置文件中,配置HttpRequestHandlerAdapter
它的作用就是适配实现了HttpRequestHandler接口的处理类,也就是找到该处理类对应的方法
如何适配,就是需要看源码了,可以百度一下讲解该适配器的源码。
SimpleControllerHandlerAdapter:适配执行实现了Controller接口的处理类的方法
在springmvc配置文件中,配置SimpleControllerHandlerAdapter
总结:这就是非注解的组件的配置方式,很简单,注意
处理器映射器和处理器适配器可以配置多个
处理器映射器和处理器适配器可以混用
4.2、配置注解的处理器映射器和适配器(掌握)
org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping 是在spring3.1之前使用的注解映射器
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping是在spring3.1之后使用的注解映射器
org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter 是在spring3.1之前使用的注解适配器
org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter是在spring3.1之后使用的注解配置器
注意:
1、注解方式的映射器和适配器在3.1版本前后是不一样的,使用3.1之后的
2、注解方式的处理器映射器和处理器适配器必须配对使用,不能与非注解的处理器映射器和适配器混用(用了注解的就不能在配置非注解的,二选一)
配置方式有两种:
1、使用bean标签配置
2、使用mvc标签(推荐)
4.3、视图解析器
4.3.1、JSP视图解析器(默认的就是使用该解析器)
其中两个配置的意思是:prefix:前缀 suffix:后缀 。 配置之后在指定视图时,就不用写这前缀和后缀了,直接写关键代码即可。看下图
虽然指定视图只写 items/itemsList 但是会帮我们加上我们配置的前缀和后缀,也就是变为了 /WEB-INF/jsp/items/itemsList.jsp
4.3.2 Freemarker视图解析器
org.springframework.web.servlet.view.freemarker.FreeMarkerViewResolver
等什么时候使用到了在学习把。
五、requestMapping注解的三种功能
requestMapping有三种功能:映射请求url、窄化请求、限制请求方法
5.1、映射请求url
也就是写在方法上,上面我们已经用过了这种功能,这里详细讲解一下
@RequestMapping(value="/item,/user")或@RequestMapping("/item”) value是数组,可以将多个url映射到同一个方法上,用逗号隔开即可。如果value中只有一个属性,则可以省去value,就像这样:@RequestMapping(value="/item ")写成@RequestMapping("/item”)
5.2、窄化请求
在class上面加上requestmapping注解,可以对url进行分类管理,这样也实现了请求的窄化
加在class上
加在方法上
访问路径为:http://localhost:8080/xxx/items/queryItems.do
5.3、限制请求方法
限制访问该方法必须是get或者post方式,相当于对请求进行过滤。
限定GET方法,也就是只能允许get请求方式过来的请求访问
@RequestMapping(method = RequestMethod.GET)
如果post请求方式的过来访问,则报错 HTTP Status 405 - Request method 'POST' not supported
限定post方法。
@RequestMapping(method = RequestMethod.POST)
如果get请求方式过来访问,报错 HTTP Status 405 - Request method 'GET' not supported
get、post都可以
@RequestMapping(method={RequestMethod.GET,RequestMethod.POST})
六、controller类中的方法返回值问题
同样有三种:ModelAndView对象、void、String
6.1、返回ModelAndView对象
controller方法中定义ModelAndView对象并返回,对象中可添加model数据、指定view。 然后通过视图解析器对其进行解析。上面用的就是这个。不用在过多的解释了
6.2、void
如果返回值为void的时候,可以在controller方法形参上定义request和response,使用request或response指定响应结果(这里在形参上定义request和response,还没讲到。但是可以这样用,相当于controller方法上默认有这两个形参。加上去就可以使用)
使用request转向页面,如下
request.getRequestDispatcher("页面路径").forward(request, response)
通过response页面重定向
response.sendRedirect("url")
可以通过response指定响应结果,例如响应json数据如下
response.setCharacterEncoding("utf-8");
response.setContentType("application/json;charset=utf-8");
response.getWriter().write("json串");
6.3、String
使用一:返回逻辑视图
解释:形参中有Model对象,该对象也是默认形参,只要声明了,就可以拿过来用,该Model对象的作用就是添加属性到request作用域中的,就跟ModelAndView对象添加值到request作用域中一样,只是model对象不能够指定视图。正好其model就是modelAndView的一半,很好理解。 其次,因为没有采用modelAndView对象,所以不能够指定视图,但是可以直接返回视图地址即可,效果是跟使用modelAndView对象一样的。
使用二:请求转发