自己实现spring核心功能 一
聊聊spring
spring对于java开发者来说,是最熟悉不过的框架了,我们日常开发中每天都在使用它。它有着各种各样的好处,简单易用,得心应手... ...
我们一说到spring就会讲到ioc 、aop、依赖注入,注解等专业名词,不少刚接触java的人,都是一头雾水,很难直观的去理解这些是个什么玩意,但使用的多了 就爱上了它给我们带来的便利。
探索spring
当我们熟练的使用它之后就会好奇,ioc怎么实现的呢?为什么我只要在类的变量中加入@AutoWrited 就能使用这个变量?带着疑惑我们就会开始翻源码,找答案。经过一番努力,最终我们定位到了spring-context.jar包,找到了
AbstractApplicationContext对象的refresh方法,这里面的实现和步骤,就是整个spring功能的核心,里面的实现另我们叹为观止,但也相当之复杂,用到了很大设计模式,难以窥见全貌,很多地方的设计我们不知道为啥需要这么实现。但我们还是知道了ioc容器实际上是用的Hashmap,
依赖注入使用的是反射,aop实际上是动态代理完成的。还有很大一部分代码看的有点晕,就是加强健壮性保证生命周期和各种特性的。不管怎么说,那都是别人的东西,我们只会用,源码我们看了很多,但很少自己来实现出来。
spring实现分析
不管怎么说,别人的实现是别人的,自己写出来的东西是属于我们自己的。既然下决心要自己也能实现了,那我们就需要分析分析了。
spring怎么做到代码侵入量少且各层级分明的呢?
spring采用约定大于配置,按照固定模式,层级分为Controller 、Service、Component、Bean、Configuration来标识类型
spring是怎么管理依赖关系的?
是通过注解或xml
spring是怎么发现哪些类需要管理的?
是注解标记+包路径扫描
spring是如何实现依赖注入的?
通过AutoWrited注解标记加反射实例化对象
spring是如何管理Bean的?
使用HashMap容器、Set容器实现单例Bean
spring怎么实现aop切面的呢?
使用动态代理的方式,并提供了cglib和jdk默认实现两种方式
spring是在何时加载到内存中的呢?
springMvc是通过web.xml配置入口触发,springboot是通过springApplication初始化触发
本次实现spring核心功能会涉及到哪些点?
1.注解的定义与使用
2.容器的初始化
3.配置文件的读取与使用
4.sevlet的使用
5.注解的定义与使用
6.反射的运用
7.url路由与方法映射
8.参数解析与绑定
9.正则与转义
正式开始
首先创建一个maven项目,代码结构如下:
添加jar包依赖:
里面有2个jar包加一个插件
1
javax.servlet-api.jar 用来启动核心代码和处理请求
1
fastjson.jar 用来做json参数绑定
jetty-maven-plugin 使用jetty作为web容器启动
1
完整maven代码如下:
View Code
创建前端控制器类
前端控制器专门处理servlet请求,匹配到对应的方法执行后返回,前端控制器是什么可以参考我的另一篇博客《SpringMvc请求处理流程与源码探秘》
这里我们创建一个叫CJDispatcherServlet的类,它继承HttpServlet类,并且重写HttpServlet的init(),doGet(),doPost() 这3个方法,图中的HomeService和StudentService可以先忽略不写
配置web.xml
需要配置 和 2个标签
中需要
1.指定servlet名称
2.指定处理请求的前端控制器类
3.设置初始化配置文件路径
完整web.xml文件如下:
View Code
添加注解
我们知道,spring里面是通过给类加注解来识别各种使用场景的,那我们就来实现几个必用的
作用在类上的:
JCController
JCService
JCComponent
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface JCController {
String value() default "";
}
全部采用这种类型
作用在方法上和类上的:
JCRequestMapping
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE,ElementType.METHOD})
public @interface JCRequestMapping {
String value() default "";
}
作用在字段上的:
JCAutoWrited
@Documented
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface JCAutoWrited {
String value() default "";
}
到此我们已经初步的创建完了需要准备的类,准备工作告一段落。
由于篇幅有限,下一篇开始实现核心功能了!