前期准备
我这里要写的是一个迷你版的Spring MVC,我将在一个干净的web工程开始开发,不引入Spring,完全通过JDK来实现。
我们先来看一眼工程:
![fcf2387a2363747bda5023d0b04fbee8.png](https://i-blog.csdnimg.cn/blog_migrate/e1ec33977b75fb5beac195f4769ced22.jpeg)
第一,在annotation包下,我将提供自定义的注解,为了方便理解,就和Spring MVC保持一致。
第二,为了模拟Spring MVC的方法调用链,我这里提供Controller/Service/Dao层进行测试
第三,提供自己的DispatcherServlet完成核心处理逻辑
关于自定义注解
JDK提供了几个元注解,比如:
- @Documented :JavaDoc文档
- @Target:标志此注解可以修饰在哪些地方,类,成员变量,方法…
- @Retention:Annotation的生命周期,一般情况下,我们自定义注解的话,显然需要在运行期获取注解的一些信息。
看一下我这边的自定义注解:
![100d4c8b9382073cccfbef9f707c5467.png](https://i-blog.csdnimg.cn/blog_migrate/28180b8f8900b54f58e6e107603ae5a9.jpeg)
模拟Spring MVC的@Controller注解
![9fa850bcfdd5fec9130313f850bdcfb5.png](https://i-blog.csdnimg.cn/blog_migrate/db250758ecba40c4ef29280560a6143b.jpeg)
@Qualifier提供依赖注入
![6ea4c354112b8a117dda4610997d8f04.png](https://i-blog.csdnimg.cn/blog_migrate/66db17e17fa53d09c52820c3ce62d10b.jpeg)
@RequestMapping提供URL地址处理映射
![431db850429173f156bcf9efbd901748.png](https://i-blog.csdnimg.cn/blog_migrate/77dc4b930f31bbd6e6fc1bef58627fd5.jpeg)
Dao层注解
![6e8143db4dc24049906fe51910332e1c.png](https://i-blog.csdnimg.cn/blog_migrate/89c6fb279156e37c128a236bdbacab58.jpeg)
Service层注解
编写核心控制器:DispatcherServlet
在Spring MVC中,DispatcherServlet是核心,下面我们来实现它。首先来说,Spring MVC中的DispatcherServlet说到底,还是HttpServlet的子类,因此我这边自己的DispatcherSerlvet需要extends HttpServlet。
pom.xml:
![433bee22fee0fd40c28a227901ac7bf7.png](https://i-blog.csdnimg.cn/blog_migrate/5307d515d159c447e8476e6b4d125080.jpeg)
看DispatcherServlet的定义:
![6297eec9618d97ed0409ab07c1002741.png](https://i-blog.csdnimg.cn/blog_migrate/394d40c15917fabf2d8fe03f619307f0.jpeg)
@WebServlet是什么?
其实,以前我们定义一个Servlet,需要在web.xml中去配置,不过在Servlet3.0后出现了基于注解的Servlet。
仔细观察,你会发现,这个DispatcherServlet是自启动,而且传入了一个参数。
要知道,在Spring MVC中,要想基于注解,需要在配置中指明扫描的包路径,就像这个样子:
为了方便,我这里就通过初始化参数直接将需要扫描的基包路径传入。
init初始化处理:
![ecab0dc1a7eb856443219dd236a2ca5f.png](https://i-blog.csdnimg.cn/blog_migrate/26525a8430658160f6b82d6b51b91a34.jpeg)
其实,在init中,我们主要是完成了什么呢?
第一,我们应该去扫描基包下的类,得到信息A
第二,对于@Controller/@Service/@Repository注解而言,我们需要拿到对应的名称,并初始化它们修饰的类,形成映射关系B
第三,我们还得扫描类中的字段,如果发现有@Qualifier的话,我们需要完成注入
第四,我们还需要扫描@RequestMapping,完成URL到某一个Controller的某一个方法上的映射关系C
其实,Spring MVC的处理流程,就是类似这样的!
扫描基包:
![1bb035c85f1ade487cf4034b953dce56.png](https://i-blog.csdnimg.cn/blog_migrate/0a5b7aeeedc5cb78b645c84d8446934b.jpeg)
注意,基包是X.Y.Z的形式,而URL是X/Y/Z的形式,需要转换。
实例化:
![314b60c81efde0a8aaf73f9089b338f7.png](https://i-blog.csdnimg.cn/blog_migrate/cad5f5c0b055bccd88f82646c1b631b8.jpeg)
从这里你可以看出,我们完成了被注解标注的类的实例化,以及和注解名称的映射。
依赖注入:
![941969d153491c89a976491378d4f812.png](https://i-blog.csdnimg.cn/blog_migrate/c824f669af643fd2ce009464d4de80a5.jpeg)
以前,我们总是说Spring IOC,上面不就是在做这个事情么?
URL映射处理:
![9b6a5c0ca791c7eaa157eb25dc2aeb5d.png](https://i-blog.csdnimg.cn/blog_migrate/116d421cdacb89aaa9e0ea403b939b1d.jpeg)
URL,我们需要提取出来,映射到Controller的Method上。
doGet/doPost:
![ecf644457a794babe50a5204113218c3.png](https://i-blog.csdnimg.cn/blog_migrate/6ef8f552205e5bcfbf73183fbad95bca.jpeg)
在doPost方法中,非常简单,我们只需要提取出URL,通过URL映射到Method上,然后通过反射的方式进行调用即可。
Make it run!
Controller层:
![4e3542d6875c48ccf9fda57fee692684.png](https://i-blog.csdnimg.cn/blog_migrate/a824eb5e169b594f6ea9a7fe7d076df2.jpeg)
Service层:
![1a1bf82fbec3ecd59b895131c596b63c.png](https://i-blog.csdnimg.cn/blog_migrate/ebd07ccd636115edb337fdf62faf873f.jpeg)
![4e889ba29108071a8442ecd7833f09f9.png](https://i-blog.csdnimg.cn/blog_migrate/f028ea6e835906ffc0c47318d8c88c7b.jpeg)
Dao层:
![a6dc2031c33f3f5ae29e3bbfba82398d.png](https://i-blog.csdnimg.cn/blog_migrate/e18bf0b8c1afea15d3522168538d69e3.jpeg)
![0fd59cc36f42d0598bee0eacf2b22116.png](https://i-blog.csdnimg.cn/blog_migrate/87f1d2e144c1423750b3a5b2c691c1f3.jpeg)
运行结果:
![09452b74bbe2f86af7a9ebff78c7bfc6.png](https://i-blog.csdnimg.cn/blog_migrate/10fe740e7c0c1e057bc6970b9b302625.jpeg)
OK,到这里,一个迷你版的Spring MVC就开发完成了,以前,都是用Spring MVC,从没有想过可以自己开发一个出来,这是我的第一次,我的能量超乎我的想象,哈哈~
Java知音,专注于Java实用文章推送,不容错过!
来源:www.jianshu.com/p/f454662f497e