首页   注册   登录
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
V2EX  ›  程序员

从 SpringBoot 到 SpringMVC

  •  
  •   hansonwang99 · 11 小时 37 分钟前 · 2024 次点击

    V A I O


    概述

    用久了 SpringBoot,深受其约定大于配置的便利性毒害之后,我想回归到 SpringMVC 时代,看看 SpringMVC 开发模式中用户是如何参与的。本文就来体验一下 SpringMVC 时代开发的流程。


    SpringMVC 架构模式

    SpringMVC 请求处理流程

    一个典型的 SpringMVC 请求流程如图所示,详细分为 12 个步骤:

    1. 用户发起请求,由前端控制器 DispatcherServlet 处理
    2. 前端控制器通过处理器映射器查找 hander,可以根据 XML 或者注解去找
    3. 处理器映射器返回执行链
    4. 前端控制器请求处理器适配器来执行 hander
    5. 处理器适配器来执行 handler
    6. 处理业务完成后,会给处理器适配器返回 ModeAndView 对象,其中有视图名称,模型数据
    7. 处理器适配器将视图名称和模型数据返回到前端控制器
    8. 前端控制器通过视图解析器来对视图进行解析
    9. 视图解析器返回真正的视图给前端控制器
    10. 前端控制器通过返回的视图和数据进行渲染
    11. 返回渲染完成的视图
    12. 将最终的视图返回给用户,产生响应

    整个过程清晰明了,下面我们将结合实际实验来理解这整个过程。


    SpringMVC 项目搭建

    实验环境如下:

    • IntelliJ IDEA 2018.1 (Ultimate Edition)
    • SpringMVC 4.3.9.RELEASE
    • Maven 3.3.9

    这里我是用 IDEA 来搭建的基于 Maven 的 SpringMVC 项目,搭建过程不再赘述,各种点击并且下一步,最终创建好的项目架构如下:

    基于 Maven 的 SpringMVC 项目


    添加前端控制器配置

    使用了 SpringMVC,则所有的请求都应该交由 SpingMVC 来管理,即要将所有符合条件的请求拦截到 SpringMVC 的专有 Servlet 上。

    为此我们需要在 web.xml 中添加 SpringMVC 的前端控制器 DispatcherServlet:

        <!--springmvc 前端控制器-->
        <servlet>
            <servlet-name>mvc-dispatcher</servlet-name>
            <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
            <init-param>
                <param-name>contextConfigLocation</param-name>
                <param-value>classpath:mvc-dispatcher.xml</param-value>
            </init-param>
        </servlet>
    
        <servlet-mapping>
            <servlet-name>mvc-dispatcher</servlet-name>
            <url-pattern>*.action</url-pattern>
        </servlet-mapping>
    

    该配置说明所有符合.action 的 url,都交由 mvc-dispatcher 这个 Servlet 来进行处理


    编写 SpringMVC 核心 XML 配置文件

    从上一步的配置可以看到,我们定义的 mvc-dispatcher Servlet 依赖于配置文件 mvc-dispatcher.xml,在本步骤中我们需要在其中添加三个方面的配置

    • 0x01. 添加处理器映射器
    <bean class="org.springframework.web.servlet.handler.BeanNameUrlHandlerMapping" />
    

    SpringMVC 的处理器映射器有多种,这里的使用的 BeanNameUrlHandlerMapping 其映射规则是将 bean 的 name 作为 url 进行处理

    • 0x02. 添加处理器适配器
    <bean class="org.springframework.web.servlet.mvc.SimpleControllerHandlerAdapter" />
    

    SpringMVC 的处理器适配器也有多种,这里的使用的 SimpleControllerHandlerAdapter 是 Controller 实现类的适配器类,其本质是执行 Controller 中的 handleRequest 方法。

    • 0x03. 添加试图解析器
    <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" />
    

    这里配置了 InternalResourceViewResolver 视图解析器后,其会根据 controller 方法执行之后返回的 ModelAndView 中的视图的具体位置,来加载对应的界面并绑定数据


    编写控制器

    这里模拟的是一个打印学生名单的 Service,我们编写的控制器需要将查询到的学生名单数据通过 ModelAndView 渲染到指定的 JSP 页面中

    public class TestController implements Controller {
    
        private StudentService studentService = new StudentService();
    
        @Override
        public ModelAndView handleRequest( HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse) throws Exception {
            List<Student> studentList = studentService.queryStudents();
            ModelAndView modelAndView = new ModelAndView();
            modelAndView.addObject("studentList",studentList);
            modelAndView.setViewName("/WEB-INF/views/studentList.jsp");
            return modelAndView;
        }
    }
    
    class StudentService {
        public List<Student> queryStudents() {
            List<Student> studentList = new ArrayList<Student>();
    
            Student hansonwang = new Student();
            hansonwang.setName("hansonwang99");
            hansonwang.setID("123456");
    
            Student codesheep = new Student();
            codesheep.setName("codesheep");
            codesheep.setID("654321");
    
            studentList.add(hansonwang);
            studentList.add(codesheep);
    
            return studentList;
        }
    }
    

    编写视图文件

    这里的视图文件是一个 jsp 文件,路径为:/WEB-INF/views/studentList.jsp

    <%@ page contentType="text/html; charset=UTF-8" pageEncoding="UTF-8" %>
    <%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c" %>
    
    <html>
    <head>
        <title>学生名单</title>
    </head>
    <body>
        <h3>学生列表</h3>
        <table width="300px;" border=1>
            <tr>
                <td>姓名</td>
                <td>学号</td>
            </tr>
            <c:forEach items="${studentList}" var="student" >
                <tr>
                    <td>${student.name}</td>
                    <td>${student.ID}</td>
                </tr>
            </c:forEach>
        </table>
    </body>
    </html>
    

    结合本步骤和上一步骤,视图和控制器都已编写完成,由于我们之前配置的处理器映射器为:BeanNameUrlHandlerMapping,因此接下来我们还需要在 mvc-dispatcher.xml 文件中配置一个可被 url 映射的 controller 的 bean,供处理器映射器 BeanNameUrlHandlerMapping 查找:

    <bean name="/test.action" class="cn.codesheep.controller.TestController" />
    

    实验测试

    启动 Tomcat 服务器,然后浏览器输入:

    http://localhost:8080/test.action
    

    实验结果

    数据渲染 OK。

    备注:当然本文所使用的全是非注解的配置方法,即需要在 XML 中进行配置并且需要遵循各种实现原则。而更加通用、主流的基于注解的配置方法将在后续文章中详述。


    后记

    作者更多的 SpringBt 实践文章在此:


    如果有兴趣,也可以抽点时间看看作者一些关于容器化、微服务化方面的文章:


    33 回复  |  直到 2018-06-11 17:29:34 +08:00
        1
    jerrry   11 小时 6 分钟前 via Android
    不错
        2
    yushiro   10 小时 20 分钟前 via iPhone
    收藏了慢慢看,正好需要
        3
    Jeffrey8888   10 小时 2 分钟前
    挺好,请楼主多多分享
        4
    SKull4   9 小时 50 分钟前
    有些错别字,建议再查一下
        5
    jatai   9 小时 48 分钟前
    想入门 java 来的,我一个渣渣看不懂,弱弱地问下:现在的趋势不是前后端分离吗,这会不会有点开倒车的?
        6
    trys1   9 小时 44 分钟前 via Android
    是该前后端分离了呀
        7
    bestkayle   9 小时 36 分钟前
    @jatai #5
    @trys1 #6 小公司不需要,大公司也不是所有都分离,前后端分离也不是什么先进的技术。
        8
    wly19960911   9 小时 35 分钟前
    @jatai #5 我认为是该前后端分离,但是有时候这些东西不应该不学,难免会碰到这种场景的需要,毕竟现在还有人用 jsp,freemarker,thymeleaf 这种模板引擎。
        9
    zqguo   9 小时 34 分钟前
    落后
        10
    sagaxu   9 小时 20 分钟前 via Android
    倒车请注意!
    @bestkayle 我一个人的项目也做前后端分离,仅仅是因为前端 mvvm 写交互界面太方便。
    @wly19960911 模板引擎不下十个,用到的时候学一下就行了,半个小时的事情。
        11
    bestkayle   9 小时 16 分钟前
    @sagaxu #10 那是你对前端非常熟的情况下,而且用那些东西对 seo 很不好,搞到最后比不分离还复杂。
        12
    grewer   9 小时 3 分钟前
    @bestkayle 但是除了 seo 现在的模式完胜以前的模板
        13
    maxiujun   8 小时 59 分钟前 via iPad
    楼主深受框架便利性毒害,应该回归纯 java,最好 servlet 也别用,tomcat 自己写个。
        14
    enzohobmg   8 小时 55 分钟前
    毒害就过分了 哈哈哈哈 恶心确实有点
        15
    liuxey   8 小时 54 分钟前
    可以再回归一下本质,用 ServerSocket 写一个 rest 服务? :doge:
        16
    lhx2008   8 小时 53 分钟前 via Android
    @maxiujun 我也是这么想的,boot 的话也就省几个配置文件而已,毒害真的说不上。如果说了解下 boot 里面的 springmvc 的运行方式也还不错。
        17
    littleghosty   8 小时 44 分钟前
    @bestkayle 小公司的 java 程序员都是前后端一把抓吗?
        18
    chinvo   8 小时 42 分钟前 via iPhone
    楼主深受 Java 便利性的毒害,赶紧汇编自己从网卡驱动写一个
        19
    bestkayle   8 小时 41 分钟前
    @littleghosty #17 PHP 基本都是的,如果没有专门的前端或者没有前端服务器不都是一把梭。
        20
    hansonwang99   8 小时 8 分钟前 via iPhone
    mark
        21
    woscaizi   8 小时 5 分钟前
    @littleghosty #17 是的
        22
    littleghosty   8 小时 4 分钟前
    @bestkayle
    @woscaizi
    哎,感觉真累啊
        23
    codeyung   8 小时 2 分钟前
    sb 也可以 用 mvc 一样的... 感觉重复工作
        24
    littleghosty   8 小时 2 分钟前
    @bestkayle 小公司还用 JSP JSTL EL ?
        25
    ljw15011154354   7 小时 59 分钟前
    前后端分类和大公司小公司有什么关系啊???现在是 2018 年不是 2008 年
        26
    pynix   7 小时 58 分钟前
    有人居然喜欢配置。。。
        27
    mringg   7 小时 58 分钟前 via iPhone
    一本正紧的开倒车
        28
    jeffson   7 小时 55 分钟前
    路过
        29
    vjnjc   7 小时 17 分钟前 via Android
    问一下 spring boot 有啥缺点么,我用了半年好像还没遇到。。。
        30
    hansonwang99   5 小时 24 分钟前 via iPhone
    滴滴滴,倒车请注意
        31
    undeflife   5 小时 0 分钟前
    看不懂在做什么
    spring boot 里对 http 请求的处理部分不还是 spring-web,spring-webmvc 在处理吗?同样可以用 jsp,ModelAndView
    如果你实在喜欢 web.xml 完全可以保留 web.xml 文件
    感觉你只是单纯的不喜欢注解,用 spring 1.x 就好了.
        32
    misaka19000   4 小时 57 分钟前
    建议楼主再写一个“从 SpringMVC 到 Servlet ”
        33
    VoidChen   1 小时 17 分钟前
    再倒一车,来个 ssh ( status+spring+hibernate )。。。手动狗头=-=
    关于   ·   FAQ   ·   API   ·   我们的愿景   ·   广告投放   ·   鸣谢   ·   2288 人在线   最高记录 3541   ·  
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.1 · 18ms · UTC 10:47 · PVG 18:47 · LAX 03:47 · JFK 06:47
    ♥ Do have faith in what you're doing.
    沪ICP备16043287号-1