看年薪50W的架构师如何手写一个SpringMVC框架(文末附视
前言
做 Java Web 开发的你,一定听说过SpringMVC的大名,作为现在运用最广泛的Java框架,它到目前为止依然保持着强大的活力和广泛的用户群。
本文介绍如何用eclipse一步一步搭建SpringMVC的最小系统,所谓最小系统,就是足以使项目在SpringMVC框架下成功跑起来,并且能够做一些简单的事情(比如访问页面)的系统。
在开始之前,想提几个问题让大家思考一下:
- 为什么要手写SpringMVC?
- 如何手写一个SpringMVC?
- 手写的SpringMVC真的可以跑起来吗?
让我们开始吧。
其他环境:
操作系统:Windos 10
Tomcat : v7.0
JDK : 1.7
正文
1. 新建一个项目
data:image/s3,"s3://crabby-images/e9134/e91342830282f30bdf06e8729f16e963b26df743" alt=""
我们用eclipse新建项目,选择Dynamic Web Project
(动态的Web项目)。
点击Next
data:image/s3,"s3://crabby-images/ef9a1/ef9a13e80ea0a4d9a7c92ad46629d174ed1c7233" alt=""
Project name
里面写上 springmvc
,这就是我们项目的名称,其他不用改,直接点击Finish
。
data:image/s3,"s3://crabby-images/65f17/65f1771ea45b75c8319588761cf5707ac53a3680" alt=""
OK,项目就建好了。
接下来一定要将项目的字符集改为UTF-8
右键项目properties
data:image/s3,"s3://crabby-images/91d72/91d72d6dd5f647d21304f598f547620b6bcfe82f" alt=""
改为UTF-8
,点击OK。
2. 编写 web.xml
当我们打开WebContent/WEB-INF
目录的时候,发现里面只有一个lib目录,这是存放各种jar包的地方。我们知道一个web项目必须要有一个web.xml
文件才行。
既然没有,我们自己写一个咯。
右键WEB-INF
——new
——file
,新建一个web.xml
文件。
点击Finish
将以下内容填进去即可。
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://xmlns.jcp.org/xml/ns/javaee"
xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd"
id="WebApp_ID">
<!-- 这是项目的名称 -->
<display-name>springmvc</display-name>
</web-app>
这样就完成了基本的配置,我的意思是说,现在这个项目就已经是一个标准的web项目了。
3. 验证web项目是否搭建成功
为了验证到目前为止的正确性,我们在WebContent
目录下面新建一个jsp
文件。
名字就叫index.jsp
data:image/s3,"s3://crabby-images/30b4b/30b4b222cb4f95d36e51ade99ce7f8b4f58be490" alt=""
内容如下:
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<html>
<head>
<meta charset="UTF-8" />
</head>
<body>
恭喜,web項目已經成功搭建!
</body>
</html>
我们现在就将这个项目部署到Tomcat
,来验证是否可以跑起来。
在项目上右键——Debug As
——Debug on Server
直接点击Finish
经过一段时间,控制台开始打印日志信息,当我们看到这些信息的时候,说明Tomcat
已经启动完毕了。
data:image/s3,"s3://crabby-images/c8b10/c8b100f7b2927896939ad33c58f21be8ea1fd8dc" alt=""
让我们打开浏览器,在地址栏输入以下信息
http://localhost:8088/springmvc/index.jsp
我电脑上Tomcat配置的端口号是8088
,具体情况视你自己的Tomcat
决定,可能是8080
等。
data:image/s3,"s3://crabby-images/96020/960203d8592f4c35152bcdf3e1c6d6634e49775a" alt=""
可见,能够成功访问页面了,这说明我们到目前为止的操作是正确的。
4. 集成SpringMVC
我们在web.xml
文件里面添加下面的配置
4.1 配置监听器
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<listener>
<listener-class>org.springframework.web.util.IntrospectorCleanupListener</listener-class>
</listener>
4.2 配置过滤器,解决POST
乱码问题
<filter>
<filter-name>encoding</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>encoding</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
4.3 配置SpringMVC
分发器,拦截所有请求
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>namespace</param-name>
<param-value>dispatcher-servlet</param-value>
</init-param>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
在这个配置中,我们规定了DispatcherServlet
的关联 XML 文件名称叫做dispatcher-servlet
。
注意,这里的路径是相对于web.xml
来说的,也就是说,这个文件也在WEB-INF
的根目录下。
所以,我们需要在WEB-INF
的根目录下新建一个dispatcher-servlet.xml
文件。
data:image/s3,"s3://crabby-images/f9e19/f9e1965111df3b744667ed1c1fb50ca1a035f685" alt=""
至此,web.xml
文件的编写就告一段落了。
4.4 编写dispatcher-servlet.xml
dispatcher-servlet.xml
的作用就是配置SpringMVC
分发器。
配置如下:
<?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:util="http://www.springframework.org/schema/util"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-3.0.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util-3.0.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc.xsd
">
<!-- 开启注解模式驱动 -->
<mvc:annotation-driven></mvc:annotation-driven>
<!-- 扫包 -->
<context:component-scan base-package="com.springmvc.*"></context:component-scan>
<!-- 静态资源过滤 -->
<mvc:resources location="/resources/" mapping="/resources/**"/>
<!-- 视图渲染 jsp/freemaker/velocity-->
<bean class="org.springframework.web.servlet.view.InternalResourceViewResolver">
<!-- 制定页面存放的路径 -->
<property name="prefix" value="/WEB-INF/pages"></property>
<!-- 文件的后缀 -->
<property name="suffix" value=".jsp"></property>
</bean>
</beans>
根据配置,有三个需要注意的地方。
-
它会扫描
com.springmvc
包下所有的Java类,但凡是遇到有注解的,比如@Controller
,@Service
,@Autowired
,就会将它们加入到Spring的bean
工厂里面去。 -
所有的静态资源文件,比如说
js
,css
,images
都需要放在/resources
目录下,这个目录现在我们还没有建。 -
所有的展示页面,比如jsp文件,都需要放置在
/WEB-INF/pages
目录下,这个目录现在我们也没有建。
OK,我们把对应的目录加上。
首先是Java
文件的目录。
data:image/s3,"s3://crabby-images/e9dbf/e9dbf6d28e9c30a5dc37c07afd7ccf5f8aa28fc9" alt=""
我们在这个地方右键,新建一个 com
包,再在里面建一个springmvc
包,或者用 .
的方式一起建。
data:image/s3,"s3://crabby-images/1e01a/1e01ab783e6ee45b8a76c856c4ccee5bbbcfca10" alt=""
点击Finish
data:image/s3,"s3://crabby-images/17b30/17b30b36ef30f29a966f72921f7917dc33120ad0" alt=""
根据SpringMVC
的分层,我们在springmvc
包下面建三个包,分别是controller
, service
, dao
data:image/s3,"s3://crabby-images/49b43/49b4305e5f59fd0760e1a07d05d959f8ce865e45" alt=""
这样的话, 当我们项目一旦启动,springmvc
就会扫描这三个包,将里面但凡是有注解的类都提取起来,放进Spring
容器(或者说Spring
的bean
工厂),借由Spring
容器来统一管理。这也就是你从来没有去new一个Controller
的原因。
接下来,我们来建静态资源的目录。
在WebContent
目录下新建一个resources
文件夹。
然后顺便把js
,css
,img
的文件夹都建一下,这里就存放我们的静态资源文件。
data:image/s3,"s3://crabby-images/fd069/fd069cf670b11528166e2babdd890ec0e9e4ce64" alt=""
最后,我们在WEB-INF
目录下建一个pages
文件夹,作为展示页面的存放目录。
data:image/s3,"s3://crabby-images/1ada1/1ada12d8be398809903c12e840346519788d6e60" alt=""
将之前的index.jsp
拷贝进来。
data:image/s3,"s3://crabby-images/c3be8/c3be88fa19c7811b13ef36a38c740b0662cb3277" alt=""
这样就配置的差不多了。
5. 导包和验证
我们将jar
包放到lib
目录:
data:image/s3,"s3://crabby-images/ee0ce/ee0ce40b862945cd445b1eff8b5babbc4f1b0f87" alt=""
然后启动项目,验证一下到目前为止的构建是否正确。
打开Servers
视图,点击如图像是甲虫一样的图标。
data:image/s3,"s3://crabby-images/20e5f/20e5f9067093129c21cfb4f24f9c7d8eafcf0cc7" alt=""
发现报错了,错误信息如下:
data:image/s3,"s3://crabby-images/31092/31092a1e2ea316b1c8020094c154845c579c7e88" alt=""
错误:
Could not open ServletContext resource [/WEB-INF/applicationContext.xml]
它说我们在WEB-INF
下面少了一个applicationContext.xml
这个文件,原来,我们少了对SpringBean
工厂的配置,它的意思就是说,我们要规定一下,在Spring
容器启动的时候,需要自动加载哪些东西?
于是,我们把 applicationContext.xml
加上。
data:image/s3,"s3://crabby-images/3f918/3f9181b7c04cc2a3f2cb57be45659a1264230502" alt=""
<?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:aop="http://www.springframework.org/schema/aop"
xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:util="http://www.springframework.org/schema/util" xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx-4.0.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.0.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util-4.0.xsd
">
</beans>
里面我们啥也不配置,再次启动Tomcat
。
data:image/s3,"s3://crabby-images/7b333/7b333f4f7935a454b6032c6e4678ab74de245bb1" alt=""
这回不报错了。
6. 配置ViewController
我们知道,WEB-INF
目录下的任何资源都是无法直接通过浏览器的url
地址去访问的,保证了安全性。这也是我们为什么把页面都放在该目录下的原因。
为了有所区分,我们还单独建立了一个pages
文件夹,将这些页面保存起来。
data:image/s3,"s3://crabby-images/953ca/953cac7fe958da380e423784252730c9299d1523" alt=""
现在,为了访问这个页面,我们需要用到SpringMVC
的页面跳转机制。
我们在Controller
包下新建一个ViewController
data:image/s3,"s3://crabby-images/8b6b8/8b6b81b5a3a93d27f28ce970d317068678b0f5d6" alt=""
点击Finish
data:image/s3,"s3://crabby-images/d6bdf/d6bdfc7e66ad8779845a40973c10c5f057cd9f04" alt=""
ViewController
代码:
package com.springmvc.controller;
import javax.servlet.http.HttpServletRequest;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;
@Controller
public class ViewController {
@RequestMapping("/view")
public ModelAndView view(HttpServletRequest request){
String path = request.getParameter("path") + "";
ModelAndView mav = new ModelAndView();
mav.setViewName(path);
return mav;
}
}
我只需要将想要访问的页面放在path
里面,通过url
传进来就行了。
因为添加了java
类,因此我们重新启动Tomcat
。
启动完成后,在地址栏输入:
http://localhost:8088/springmvc/view?path=index
结果:
data:image/s3,"s3://crabby-images/b08c5/b08c5bc999871a248b593dc37ab6142e3c92ca16" alt=""
没关系,我们看他报什么错。
message /springmvc/WEB-INF/pagesindex.jsp
pagesindex.jsp
是什么鬼??
原来,在dispatcher-servlet.xml
中,我们少写了一个 "/"
data:image/s3,"s3://crabby-images/c64d1/c64d118ac3f603b21e78b1d3832c095418b8c30f" alt=""
添上去就行了。
data:image/s3,"s3://crabby-images/23b8c/23b8c9bf730164dea3a7d9aac9af447c55a406ef" alt=""
保存后,因为修改了XML
配置文件,因此我们还是需要重新启动Tomcat
。
启动完成后,继续!
data:image/s3,"s3://crabby-images/ef7f8/ef7f85bed26fa0138cf80346d5083466e5c8aa3a" alt=""
成功了。
7. 引入静态资源
比如,我在resources/img
目录下放了一张图片,怎么引入到index.jsp
呢?
data:image/s3,"s3://crabby-images/efbcb/efbcbb8097a4460822f336cecf0feb72b39b192e" alt=""
background : url(http://localhost:8088/springmvc/resources/img/bg.jpg);
background-size : 100% 100%;
的确,这是一种方式。可是,它有一个缺点就是根路径写死了,我们肯定不希望这样的。
其实,我们可以在viewController
里面拿到项目根路径,然后传递到jsp
页面就OK了。
data:image/s3,"s3://crabby-images/6535f/6535fc987e4d556bf4378e4dfa65185d516c99de" alt=""
我们把调试信息 “恭喜,web項目已經成功搭建!” 删掉。
<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<html>
<head>
<meta charset="UTF-8" />
</head>
<style>
body {
background : url(${contextPath}/resources/img/bg.jpg);
background-size : 100% 100%;
}
</style>
<body>
</body>
</html>
${contextPath}
可以取到Controller
传过来的contextPath
值。
成功了!
真正的干货
珍贵的视频:《用400行代码手写一个SpringMVC框架》——完全可用
主要内容:
- 一小时写完核心代码,重温经典,不复制、不粘贴。
- 用400行代码浓缩思想精华,麻雀虽小,五脏俱全。
- 带你体验IOC、DI、MVC从0到1发生的故事。
- 死记硬背容易忘却,学习方法终身受益。
学习永不止步,本视频原文件放在群:895244712
里,群内还有分布式,微服务,性能优化等技术点底层原理的录像视频,欢迎大家加群一起交流学习。
最后,如有帮助请转发,点赞。您的拇指是对我最好的支持!