美文网首页Java
重温SpringMVC:框架原理解读 + 简单入门程序+组件分析

重温SpringMVC:框架原理解读 + 简单入门程序+组件分析

作者: 废柴程序员 | 来源:发表于2021-07-14 15:59 被阅读0次

    一、什么是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的运用概念图

    image

    二、springmvc的入门程序

    通过这个来快速了解springmvc大概的开发流程,其实通过上面的mvc分析图,差不多就知道了如何开发了。重点就是三步。

    • 1、在web.xml中配置一个serlvet,用来控制,
    • 2、编写一个handler(controller)类,用来做业务处理。
    • 3、编写jsp或者别的视图,用来展示数据

    思路已经有了,那么就开始编写吧。

    问题描述:使用springmvc来完成前端请求的处理

    2.1、创建web工程

    image

    2.2、添加jar包

    image

    2.3、编程步骤

    前面三步只是通过mvc图的分析出最关键的三步,其中实现的时候步骤应该更多,比如spring的配置文件,但关键的重点还是那三个。

    • 1、创建po类
    • 2、配置前端控制器,DispatcherServlet
    • 3、创建springmvc的配置文件
    • 4、开发handler(controller)
    • 5、在springmvc的配置文件中(取名为springmvc.xml)配置handler
    • 6、开发jsp或者别的视图
    • 8、部署测试

    2.4、创建po类

    image image

    2.5、配置前端控制器

    image
     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文件

    image
    <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接口

    image
     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,也就能够找到)

    image
    1         <!-- 配置实现HttpRequestHander接口的处理器 -->
    2         <bean name="/queryItems01.do" class="com.wuhao.springmvc.controller.ItemController"></bean>
    

    实现Controller接口

    image
     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类

    image
            <!-- 配置实现Controller接口的处理器 -->
            <bean name="/queryItems02.do" class="com.wuhao.springmvc.controller.ItemController02"></bean>
    

    使用注解开发

    image

    注解的配置,就是配置一个扫描器,扫描使用了注解的地方

    image
            <!-- 使用注解的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

    image
    <%@ 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如何查找到视图对象的?

    看图即可

    image
    • 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中的三个重点,在第二小节中就是这样编写的,第三小结就是解释其中的原理。

    处理器映射器、处理器适配器、视图解析器这三个是默认配置的,在下面位置中可以查看

    image

    DispatchServlet.properties

    image

    BeanNameUrlHandlerMapping:映射器

    4.1、非注解的处理器映射器和处理器适配器  [看看即可]

    BeanNameUrlHandlerMapping:映射器

    在springmvc配置文件中,配置BeanNameUrlHandlerMapping

    image

    他的作用是找到在springmvc.xml中配置的url和处理器的bean

    image

    HttpRequestHandlerAdapter:适配执行实现了HttpRequestHandler接口的处理类的方法

    在springmvc配置文件中,配置HttpRequestHandlerAdapter

    image

    它的作用就是适配实现了HttpRequestHandler接口的处理类,也就是找到该处理类对应的方法

    如何适配,就是需要看源码了,可以百度一下讲解该适配器的源码。

    SimpleControllerHandlerAdapter:适配执行实现了Controller接口的处理类的方法

    在springmvc配置文件中,配置SimpleControllerHandlerAdapter

    image

    总结:这就是非注解的组件的配置方式,很简单,注意

    • 处理器映射器和处理器适配器可以配置多个
    • 处理器映射器和处理器适配器可以混用

    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标签配置

    image

    2、使用mvc标签(推荐)

    image

    4.3、视图解析器

    4.3.1、JSP视图解析器(默认的就是使用该解析器)

    image

    其中两个配置的意思是:prefix:前缀 suffix:后缀 。 配置之后在指定视图时,就不用写这前缀和后缀了,直接写关键代码即可。看下图

    image

    虽然指定视图只写 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上

    image

    加在方法上

    image

    访问路径为:http://localhost:8080/xxx/items/queryItems.do

    5.3、限制请求方法

    限制访问该方法必须是get或者post方式,相当于对请求进行过滤。

    限定GET方法,也就是只能允许get请求方式过来的请求访问

    @RequestMapping(method = RequestMethod.GET)

    image

    如果post请求方式的过来访问,则报错 HTTP Status 405 - Request method 'POST' not supported

    限定post方法。

    @RequestMapping(method = RequestMethod.POST)

    image

    可以通过response指定响应结果,例如响应json数据如下

    get、post都可以@RequestMapping(method={RequestMethod.GET,RequestMethod.POST})

    image

    六、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

    使用一:返回逻辑视图

    image

    解释:形参中有Model对象,该对象也是默认形参,只要声明了,就可以拿过来用,该Model对象的作用就是添加属性到request作用域中的,就跟ModelAndView对象添加值到request作用域中一样,只是model对象不能够指定视图。正好其model就是modelAndView的一半,很好理解。 其次,因为没有采用modelAndView对象,所以不能够指定视图,但是可以直接返回视图地址即可,效果是跟使用modelAndView对象一样的。

    使用二:请求转发

    image

    相关文章

      网友评论

        本文标题:重温SpringMVC:框架原理解读 + 简单入门程序+组件分析

        本文链接:https://www.haomeiwen.com/subject/qpmypltx.html