SpringMVC 应知应会
springMVC 是表现层技术,可以用来代替 struts2,下面是简略图:主要是处理器和视图,只有这两个部分需要编写代码。
springMVC 三大组件:处理器映射器,处理器适配器,视图解析器。
下面是 SpringMVC的细节图:
整个流程:
- 客户端的请求到达前端控制器 DispatcherServlet
- DispatcherServlet 收到请求后调用 HandlerMapping 处理器映射器
- 处理器映射器根据请求的 url 找到具体的处理器,生成处理器对象以及处理器拦截器【不一定有,但是有的话一定会生成】,并将生成的对象交给 DispatcherServlet
- DispatcherServlet 接收到处理器映射器反馈的信息,通过 HandlerAdapter 处理器适配器找到对应的处理器
- 执行处理器的逻辑【处理器也叫后端控制器,Controller 】
- Controller 执行完毕后返回 ModelAndView 对象【也可以返回 String】
- 再次回到 HandlerAdapter,HandlerAdapter 将 Controller 的执行结果 ModelAndView 返回給 DispatcherServlet,
- DispatcherServlet 接收到 ModelAndView 之后,调用 ViewReslover 视图解析器
- ViewReslover 解析猴年返回具体的 View 对象
- DispatcherServlet 对返回回来的 View 对象进行渲染【即将模型数据填充至视图】
- DispatcherServlet 响应客户端的请求
上述的十一个步骤,其中需要编写代码的只有处理器【Handler 即 Controller】和 视图【View】,其他的只需要配置即可。
处理器的编写:
1 . 处理器类需要添加配置成 bean,因为是在表现层,所有使用的是 @Controller注解,
2 . 处理器中的方法需要使用 @RequestMapping(“url”)注解
3 . 处理器中方法的形参列表支持的几个特殊的对象:HttpServletRequest,HttpServletResponse,HttpServletSession,
见名知意,前端控制器会自动将这三个对象进行封装,便于方法内部获取请求参数和设置参数。
4 . 参数列表还支持一个参数——Modle【这个类是个接口,其子类 ModelMap】 ,但是它不是用来获取请求参数,
而是通过它向 request域存储数据,向视图传递数据【视图可以理解为 jsp,但视图不只是 jsp】,
5 . 处理器类中方法支持的返回值类型有:
- ModelAndView —— 携带了数据和要响应的视图【数据存储在 request域中】
- String —— 携带了响应的视图,一般结合 Model形参使用,一个携带响应的视图,一个携带数据【数据存储在 request域中】
- void —— 一般用于处理 AJAX 的请求,返回的数据由 Model形参来携带数据【或者其子类 ModelMap】,然后使用 request的转发或者 response进行重定向
小小总结:使用 ModelAndView,Model,ModelMap携带数据都是存储在了 request域中了,【 此时请求还未结束,也就是request对象还存活着 】,在 jsp页面使用 JSTL即可获取数据。
例子:
@Controller
public class ItemController { @Autowired
private ItemService itemService; /**
* 显示商品列表
*
* @return
*/
@RequestMapping("/itemEdit")
public String queryItemById(HttpServletRequest request, ModelMap model) {
// 从request中获取请求参数
String strId = request.getParameter("id");
Integer id = Integer.valueOf(strId); // 根据id查询商品数据
Item item = this.itemService.queryItemById(id); // 第一种方式:使用 ModelAndView ----------------------------
// 把结果传递给页面
// ModelAndView modelAndView = new ModelAndView();
// 把商品数据放在模型中
// modelAndView.addObject("item", item);
// 设置逻辑视图
// modelAndView.setViewName("itemEdit");
// 返回携带数据和视图的 ModelAndView对象
// return modelAndView;
// 第二种方式:使用 Model --------------------------------------
// 把商品数据放在模型中,其实是放在了 request域中
model.addAttribute("item", item);
// 返回视图的名字,也就是 jsp的名字【jsp视图的路径前缀和后缀需在 springmvc.xml中配置】
return "itemEdit";
}
}
上面代码中写到的视图路径前缀和后缀的配置如下:【也就是配置视图解析器】
<!-- 配置视图解析器 -->
<bean
class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<!-- 配置逻辑视图的前缀 -->
<property name="prefix" value="/WEB-INF/jsp/" />
<!-- 配置逻辑视图的后缀 -->
<property name="suffix" value=".jsp" />
</bean>
需要的配置:
1 . 添加 springmvc.xml文件,配置包扫描器,用于扫描 Controller
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:p="http://www.springframework.org/schema/p"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc-4.0.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd"> <!-- 配置controller扫描包,多个包之间用,分隔 -->
<context:component-scan base-package="com.msym.springmvc.controller" /> </beans>2 . 配置前端控制器,在 web.xml 中配置 DispatcherServlet
<?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"
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-first</display-name>
<!-- 配置 SpringMVC 前端控制器 -->
<servlet>
<servlet-name>springmvc-first</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<!-- 指定 SpringMVC 配置文件 -->
<!-- SpringMVC 的配置文件的默认路径是 /WEB-INF/${servlet-name}-servlet.xml -->
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc.xml</param-value>
</init-param>
</servlet> <servlet-mapping>
<servlet-name>springmvc-first</servlet-name>
<!-- 设置所有以 action结尾的请求进入 SpringMVC -->
<url-pattern>*.action</url-pattern>
</servlet-mapping>
</web-app>3 . 创建 Controller ,Controller只是一个普通 Java 类,需要使用 @Controller 进行注解,其中的方法使用 @RequestMapping 进行注解
4 . 在 springmvc.xml 中配置处理器映射器,处理器适配器,视图解析器【下面的第五步可以替代这一步】
<!-- 配置处理器映射器 -->
<bean
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping" />
<!-- 配置处理器适配器 -->
<bean
class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter" />5 . 可以采用注解驱动的方式,代替了处理器映射器,处理器适配器的配置。【第五步可以完全替代第四步】
<!-- 注解驱动 -->
<mvc:annotation-driven />
SpringMVC 的拦截器:
Spring Web MVC 的处理器拦截器类似于Servlet 开发中的过滤器Filter,用于对处理器进行预处理和后处理。
之所以有预处理和后处理,是因为拦截器在处理器前后都会执行,只不过执行的方法不一样。
自定义拦截器需要实现 HandlerIntercept 接口,下面是【爱上笔记】中的一个拦截器类:
package cn.evelynn.cloudnote.interceptors; import java.io.PrintWriter; import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession; import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView; import com.fasterxml.jackson.databind.ObjectMapper; import cn.evelynn.cloudnote.entity.User;
import cn.evelynn.cloudnote.util.JsonResult; /**
* 拦截器:用于拦截未登录的用户
* @author 码上猿梦
* http://www.cnblogs.com/daimajun/
*/
@Component
public class AccessInterceptor implements HandlerInterceptor { /**
* 用于判断用户是否登录,
* 未登录则拦截,
* 已登录就放行
*/
public boolean preHandle(HttpServletRequest req, HttpServletResponse res, Object obj) throws Exception {
HttpSession session = req.getSession();
//判断用户是否登录
User user = (User) session.getAttribute("user");
if (user == null) {
// 利用response对象返回结果,告诉浏览器是Json格式,编码为UTF-8
res.setContentType("application/json;charset=UTF-8");
PrintWriter out = res.getWriter();
//ObjectMapper能将java对象转换为匹配Json的结构
ObjectMapper om = new ObjectMapper();
//将字符串转换为Json格式的字符串
String json = om.writeValueAsString(new JsonResult("请重新登录!"));
out.write(json);
//刷新缓冲区,向浏览器写出数据
res.flushBuffer();
return false;
}
//放行
return true;
} // controller执行后但未返回视图前调用此方法
// 这里可在返回用户前对模型数据进行加工处理,比如这里加入公用信息以便页面显示
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
// TODO Auto-generated method stub } // controller执行后且视图返回后调用此方法
// 这里可得到执行controller时的异常信息
// 这里可记录操作日志
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
throws Exception {
// TODO Auto-generated method stub }
}
然后就是配置拦截器了,需要在 springmvc.xml 文件中配置拦截器:
<!-- 配置Interceptor -->
<mvc:interceptors>
<mvc:interceptor>
<!-- 请求 /note/ 和 /notebook/ 资源的请求都进入拦截器 -->
<mvc:mapping path="/note/*" />
<mvc:mapping path="/notebook/*" />
<!-- AccessInterceptor 类采用了注解注册bean -->
<ref bean="accessInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
自定义异常类:
SpringMVC 应知应会的更多相关文章
- 测试TwemProxy的应知应会
一.背景 最近中间件开发组对twemproxy的发现注册机制做了改造,之前没有接触过twemproxy,借这次测试的机会,初步学习了一下twemproxy相关的知识:下面用"测试语言&quo ...
- Markdown的应知应会
Markdown介绍 什么是Markdown Markdown是一种纯文本.轻量级的标记语言,常用作文本编辑器使用.和记事本.notepad++相比,Markdown可以进行排版:和Word相比,Ma ...
- Linux用户应知应会的7个‘ls’命令的独特技巧
在前面我们系列报道的两篇文章中,我们已经涵盖了关于‘ls’命令的绝大多数内容.本文时‘ls命令’系列的最后一部分.如果你还没有读过该系列的其它两篇文章,你可以访问下面的链接. Linux中的15个基本 ...
- 【应知应会】15个常用的JavaScript字符串操作方法
1 初始化 //常用初始化方法 var stringVal = "hello iFat3"; //构造函数创建方法 var stringObj = new String(" ...
- Hibernate 应知应会
Hibernate 的关联关系的配置: 一对一外键约束: 举例子是一个丈夫和妻子:[一个丈夫只能有一位妻子] 表结构: CREATE TABLE `tbl_hus` ( `uuid` ) NOT NU ...
- Struts2 应知应会
struts.xml 文件的 action 的配置: Struts2 中结果类型的配置来自于下面: 其中: dispatcher:转发技术,转发到一个 jsp 视图 redirect:重定向到一个 j ...
- .NET架构开发应知应会
.NET程序是基于.NET framework..NET Core.Mono.UWP[.NET实现]开发和运行的 ,定义以上[.NET实现]的标准规范称为.NET Standard L1:.NET S ...
- Java 多线程应知应会
请简单说说 synchronized 关键字的底层原理 java 说到多线程绝对绕不开 synchronized,很多 java 工程师对 synchronized 是又爱又恨.为什么呢?主要原因包括 ...
- 关于HDFS应知应会的N个问题 | 技术点
1. Namenode的安全模式 ? 安全模式是Namenode的一种状态(Namenode主要有active/standby/safemode三种模式). 2. 哪些情况下,Namenode会进入安 ...
随机推荐
- FPGA时序逻辑中常见的几类延时与时间(五)
FPGA逻辑代码重要的是理解其中的时序逻辑,延时与各种时间的记忆也是一件头疼的事,这里把我最近看到的比较简单的几类总结起来,共同学习. 一.平均传输延时 平均传输延时 二.开启时间与关闭时间 开 ...
- [Java算法分析与设计]--链式堆栈的设计
在上篇文章当中,我们实现了底层为数组的顺序栈.在我之前的文章中也提到过:以数组为数据结构基础在存储数据方面需要一整块连续的内存来存放数据,一旦遇到需要可以动态扩展的功能需求时如果数据量大可能会给虚拟机 ...
- Eclipse怎么恢复默认界面
Eclipse里面将界面恢复到默认状态: 1.选择Eclipse的工具栏里面的“窗口(Window)”,找到“复位透视图(Reset Perspective)”选项: 2.单击“复位透视图(Reset ...
- WPF几个基础概念的浅显理解
1.逻辑树与视觉树 逻辑树在结构上与xaml文件对应 视觉树更细化,拆分到控件的每个组成部分 2.依赖属性与附加属性 依赖属性:就是自己自己没有属性值,而是通过Binding从数据源获得值,就是依赖在 ...
- 20155218 《Java程序设计》实验一(Java开发环境的熟悉)实验报告
20155218 <Java程序设计>实验一(Java开发环境的熟悉)实验报告 一.实验内容及步骤 (一)使用JDK编译.运行简单的java程序 实验结果截图: (二)使用IDEA编辑.编 ...
- [agc006F]Blackout
Description 传送门 Solution 这道题的操作是真的得服气..感谢各位大佬的指导. 首先我们看看答案的最大值:1010.哦不,这不可能存在,我们肯定不可能一轮轮枚举点进行扩展的. 所以 ...
- centos7下python3与python2共存并且开启py3虚拟环境
因为下载视频需要用到python3环境,今天在我的win上安装下载工具死活安装不上去,在大盘鸡上一下就安装成功了...可能在win上不兼容吧...无奈只能在大盘鸡上进行折腾了,顺便几个笔记 由于大盘鸡 ...
- 行驶证识别/行驶证OCR识别全方位解析
本文全面解析行驶证OCR识别,包括什么是行驶证OCR识别.如何选择行驶证识别软件.如何操作行驶证识别软件,以及该软件应用的领域等. 一.了解行驶证识别/行驶证OCR识别 行驶证OCR识别技术,也叫行驶 ...
- linux命令提示符设置
变成绿色 [root@localhost /usr/local]#vim /root/.bashrc # .bashrc # User specific aliases and functions a ...
- Lua学习笔记(5): 表
表的初始化方式 表的索引类型一般有两种,一种是通过标识符访问,一种是通过数字访问 --通过标识符访问的表的初始化 table1 = {key_1 = "haha", key_2 = ...