前期准备
我这里要写的是一个迷你版的Spring MVC,我将在一个干净的web工程开始开发,不引入Spring,完全通过JDK来实现。
我们先来看一眼工程:
工程代码结构
第一,在annotation包下,我将提供自定义的注解,为了方便理解,就和Spring MVC保持一致。
第二,为了模拟Spring MVC的方法调用链,我这里提供Controller/Service/Dao层进行测试
第三,提供自己的DispatcherServlet完成核心处理逻辑
关于自定义注解
JDK提供了几个元注解,比如:
@Documented : JavaDoc文档
@Target:标志此注解可以修饰在哪些地方,类,成员变量,方法...
@Retention:Annotation的生命周期,一般情况下,我们自定义注解的话,显然需要在运行期获取注解的一些信息。
看一下我这边的自定义注解:
模拟Spring MVC的@Controller注解
@Qualifier提供依赖注入
@RequestMapping提供URL地址处理映射
Dao层注解
Service层注解
编写核心控制器:DispatcherServlet
在Spring MVC中,DispatcherServlet是核心,下面我们来实现它。首先来说,Spring MVC中的DispatcherServlet说到底,还是HttpServlet的子类,因此我这边自己的DispatcherSerlvet需要extends HttpServlet。
pom.xml:
提供servlet依赖
看DispatcherServlet的定义:
DispatcherServlet
@WebServlet是什么?
其实,以前我们定义一个Servlet,需要在web.xml中去配置,不过在Servlet3.0后出现了基于注解的Servlet。
仔细观察,你会发现,这个DispatcherServlet是自启动,而且传入了一个参数。
要知道,在Spring MVC中,要想基于注解,需要在配置中指明扫描的包路径,就像这个样子:
<context:component-scan base-package="com.zfz.myspringmvc">
</context:component-scan>
为了方便,我这里就通过初始化参数直接将需要扫描的基包路径传入。
init初始化处理:
init()
其实,在init中,我们主要是完成了什么呢?
第一,我们应该去扫描基包下的类,得到信息A
第二,对于@Controller/@Service/@Repository注解而言,我们需要拿到对应的名称,并初始化它们修饰的类,形成映射关系B
第三,我们还得扫描类中的字段,如果发现有@Qualifier的话,我们需要完成注入
第四,我们还需要扫描@RequestMapping,完成URL到某一个Controller的某一个方法上的映射关系C
其实,Spring MVC的处理流程,就是类似这样的!
扫描基包:
扫描基包
注意,基包是X.Y.Z的形式,而URL是X/Y/Z的形式,需要转换。
实例化:
实例化
从这里你可以看出,我们完成了被注解标注的类的实例化,以及和注解名称的映射。
依赖注入:
依赖注入
以前,我们总是说Spring IOC,上面不就是在做这个事情么?
URL映射处理:
URL映射处理
URL,我们需要提取出来,映射到Controller的Method上。
doGet/doPost:
doGet/doPost
在doPost方法中,非常简单,我们只需要提取出URL,通过URL映射到Method上,然后通过反射的方式进行调用即可。
Make it run!
Controller层:
UserController
Service层:
UserService
UserServiceImpl
Dao层:
UserDao
UserDaoImpl
运行结果:
result
OK,到这里,一个迷你版的Spring MVC就开发完成了,以前,都是用Spring MVC,从没有想过可以自己开发一个出来,这是我的第一次,我的能量超乎我的想象,哈哈~
在此我向大家推荐一个架构资料,马士兵JVM精讲视频,威信,an_sheng975里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多