Spring学习之——手写Mini版Spring源码
前言
Sping的生态圈已经非常大了,很多时候对Spring的理解都是在会用的阶段,想要理解其设计思想却无从下手。前些天看了某某学院的关于Spring学习的相关视频,有几篇讲到手写Spring源码,感觉有些地方还是说的挺好的,让博主对Spring的理解又多了一些,于是在业余时间也按照视频讲解实现一遍SpringIOC、DI、MVC的设计思想,加深巩固记忆,也便于初学者理解,我们不用重复造轮子,但得知道轮子怎么造。
开发工具
环境:jdk8 + IDEA + maven
jar包:javax.servlet-2.5
实现步骤

视频中这张图画得很好,我们按照这张图来概括一下实现步骤
1.配置阶段:即配置web.xml和application.properties,以及相关自定义注解;
2.初始化阶段:初始化Ioc容器,将扫描到的类实例化交给IoC容器管理,然后注入实例的各属性,接着将请求路径与执行方法的映射加载到HandlerMapping中;
3.运行阶段:由HandlerMapping根据请求路径将请求分发到指定方法,返回执行结果。
附上自己的项目结构供参考

具体实现
配置阶段
application.properties 配置扫描路径,这里为了方便,所以没有使用xml文件去解析
scanPackage=com.wqfrw
web.xml 配置Servlet和application文件的路径
<servlet>
<servlet-name>mymvc</servlet-name>
<servlet-class>com.framework.servlet.MyDispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>application.properties</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet> <servlet-mapping>
<servlet-name>mymvc</servlet-name>
<url-pattern>/*</url-pattern>
</servlet-mapping>
pom.xml 引入servlet.jar
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
相关自定义注解
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyAutowired {
String value() default "";
} @Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyController {
String value() default "";
} @Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyRequestMapping {
String value() default "";
} @Target(ElementType.PARAMETER)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyRequestParam {
String value() default "";
} @Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyService {
String value() default "";
}
初始化阶段
这部分的操作才是最重要的,注释我都详细的写在代码中
MyDispatcherServlet
public class MyDispatcherServlet extends HttpServlet {
//IoC容器
private Map<String, Object> ioc = new HashMap<>();
//加载配置文件对象
private Properties contextConfig = new Properties();
//扫描到的类名带包路径
private List<String> classNameList = new ArrayList<>();
//url与方法的映射 也就是请求分发器
private Map<String, Method> handlerMapping = new HashMap<>();
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
doPost(req, resp);
}
@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
try {
doDispatch(req, resp);
} catch (Exception e) {
e.printStackTrace();
resp.getWriter().write("500 Exception Detail:" + Arrays.toString(e.getStackTrace()));
}
}
@Override
public void init(ServletConfig config) throws ServletException {
//加载配置文件 拿到需要扫描的包路径
doLoadConfig(config.getInitParameter("contextConfigLocation"));
//扫描相关的类
doScanner(contextConfig.getProperty("scanPackage"));
//实例化Bean到IoC容器
doInstance();
//依赖注入(DI)
doAutowired();
//初始化HandlerMapping url和对应方法的键值对
doInitHandlerMapping();
//初始化完成
System.out.println("MySpring framework is init!");
}
/**
* 功能描述: 接收到浏览器的请求,执行方法
*
* @创建人: 我恰芙蓉王
* @创建时间: 2020年07月28日 20:54:04
* @param req
* @param resp
* @return: void
**/
private void doDispatch(HttpServletRequest req, HttpServletResponse resp) throws Exception {
String url = req.getRequestURI();
String contextPath = req.getContextPath();
url = url.replaceAll(contextPath,"");
//根据请求路径如果没有找到对应的方法则抛出404错误
if (!handlerMapping.containsKey(url)) {
resp.getWriter().write("404 Not Found!");
return;
}
Map<String,String[]> params = req.getParameterMap();
Method method = handlerMapping.get(url);
String beanName = toLowerFirstCase(method.getDeclaringClass().getSimpleName());
//方法调用
method.invoke(ioc.get(beanName),new Object[]{req,resp,params.get("name")[0]});
}
/**
* 功能描述: 初始化HandlerMapping
*
* @创建人: 我恰芙蓉王
* @创建时间: 2020年07月28日 20:56:09
* @param
* @return: void
**/
private void doInitHandlerMapping() {
ioc.forEach((k, v) -> {
Class<?> clazz = v.getClass();
//加了MyController注解的类才操作
if (clazz.isAnnotationPresent(MyController.class)) {
String baseUrl = "";
//如果类上面加了MyRequestMapping注解,则需要拿到url进行拼接
if (clazz.isAnnotationPresent(MyRequestMapping.class)) {
MyRequestMapping annotation = clazz.getAnnotation(MyRequestMapping.class);
baseUrl = annotation.value();
}
//获取所有public修饰的方法
Method[] methods = clazz.getMethods();
//过滤拿到所有MyRequestMapping注解的方法,put到handlerMapping中
String finalBaseUrl = baseUrl;
Stream.of(methods)
.filter(m -> m.isAnnotationPresent(MyRequestMapping.class))
.forEach(m -> {
MyRequestMapping annotation = m.getAnnotation(MyRequestMapping.class);
String url = (finalBaseUrl + annotation.value()).replaceAll("/+", "/");
handlerMapping.put(url, m);
});
}
});
}
/**
* 功能描述: 依赖注入(DI)
*
* @创建人: 我恰芙蓉王
* @创建时间: 2020年07月28日 20:55:57
* @param
* @return: void
**/
private void doAutowired() {
//循环IoC容器中所管理的对象 注入属性
ioc.forEach((k, v) -> {
//拿到bean所有的字段 包括private、public、protected、default
Field[] Fields = v.getClass().getDeclaredFields();
//过滤拿到所有加了MyAutowired注解的字段并循环注入
Stream.of(Fields)
.filter(f -> f.isAnnotationPresent(MyAutowired.class))
.forEach(f -> {
MyAutowired annotation = f.getAnnotation(MyAutowired.class);
String beanName = annotation.value().trim();
if ("".equals(beanName)) {
beanName = f.getType().getName();
}
//强制访问
f.setAccessible(true);
try {
//赋值
f.set(v, ioc.get(beanName));
} catch (IllegalAccessException e) {
e.printStackTrace();
}
});
});
}
/**
* 功能描述: 实例化bean至IoC容器
*
* @创建人: 我恰芙蓉王
* @创建时间: 2020年07月28日 20:55:39
* @param
* @return: void
**/
private void doInstance() {
classNameList.forEach(v -> {
try {
Class<?> clazz = Class.forName(v);
//只初始化加了MyController注解和MyService注解的类
if (clazz.isAnnotationPresent(MyController.class)) {
String beanName = toLowerFirstCase(clazz.getSimpleName());
ioc.put(beanName, clazz.newInstance());
} else if (clazz.isAnnotationPresent(MyService.class)) {
// 1.默认首字母小写
String beanName = toLowerFirstCase(clazz.getSimpleName());
// 2.自定义beanName
MyService myService = clazz.getAnnotation(MyService.class);
if (!"".equals(myService.value())) {
beanName = myService.value();
}
// 3.如果是接口 必须创建实现类的实例
for (Class<?> i : clazz.getInterfaces()) {
if (ioc.containsKey(i)) {
throw new Exception("This beanName is exists!");
}
beanName = i.getName();
}
//将实例化的对象放入IoC容器中
ioc.put(beanName, clazz.newInstance());
}
} catch (Exception e) {
e.printStackTrace();
}
});
}
/**
* 功能描述: 扫描相关的类 加入到classNameList
*
* @创建人: 我恰芙蓉王
* @创建时间: 2020年07月28日 20:55:10
* @param scanPackage
* @return: void
**/
private void doScanner(String scanPackage) {
//获取根目录 拿到com.wqfry替换成/com/wqfrw
URL url = this.getClass().getClassLoader().getResource("/" + scanPackage.replaceAll("\\.", "/"));
File classFile = new File(url.getFile());
for (File file : classFile.listFiles()) {
//如果file是文件夹 则递归调用
if (file.isDirectory()) {
doScanner(scanPackage + "." + file.getName());
} else {
//如果非class文件 则跳过
if (!file.getName().endsWith(".class")) {
continue;
}
String className = (scanPackage + "." + file.getName()).replace(".class", "");
//类名+包路径放入到类名集合中 方便后续实例化
classNameList.add(className);
}
}
}
/**
* 功能描述: 加载配置文件
*
* @创建人: 我恰芙蓉王
* @创建时间: 2020年07月28日 20:54:57
* @param contextConfigLocation
* @return: void
**/
private void doLoadConfig(String contextConfigLocation) {
InputStream resource = this.getClass().getClassLoader().getResourceAsStream(contextConfigLocation);
try {
//加载配置文件
contextConfig.load(resource);
} catch (IOException e) {
e.printStackTrace();
} finally {
//关闭文件流
if (resource != null) {
try {
resource.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
}
/**
* 首字母转小写
*
* @param className
* @return
*/
private String toLowerFirstCase(String className) {
char[] chars = className.toCharArray();
chars[0] += 32;
return String.valueOf(chars);
}
}
运行阶段
部分代码在上面,这里我贴出Controller与Server的代码
TestController
@MyController
@MyRequestMapping("/test")
public class TestController { @MyAutowired
private ITestService testService; @MyRequestMapping("/query")
public void query(HttpServletRequest req, HttpServletResponse resp, @MyRequestParam("name") String name){
String result = testService.query(name);
try {
resp.getWriter().write(result);
} catch (IOException e) {
e.printStackTrace();
}
} @MyRequestMapping("/add")
public void add(HttpServletRequest req, HttpServletResponse resp, @MyRequestParam("name") String name){
System.out.println("add");
} @MyRequestMapping("/remove")
public void remove(HttpServletRequest req, HttpServletResponse resp, @MyRequestParam("name") String name){
System.out.println("remove");
} }
ITestService
public interface ITestService {
String query(String name);
}
TestServiceImpl
@MyService
public class TestServiceImpl implements ITestService { @Override
public String query(String name) {
return "hello " + name + "!";
} }
实际调用

总结
以上只是简略实现了Spring的核心思想,真正的Spring当然要比此复杂许多,但是学习都是由浅至深的,希望大家不仅会用工具,并且都能知道为什么要这样用。
附上原视频链接:https://live.gupaoedu.com/watch/1284875
Spring学习之——手写Mini版Spring源码的更多相关文章
- Spring学习之——手写Spring源码V2.0(实现IOC、D、MVC、AOP)
前言 在上一篇<Spring学习之——手写Spring源码(V1.0)>中,我实现了一个Mini版本的Spring框架,在这几天,博主又看了不少关于Spring源码解析的视频,受益匪浅,也 ...
- 手写mini版MVC框架
目录 1, Springmvc基本原理流程 2,注解开发 编写测试代码: 目录结构: 3,编写自定义DispatcherServlet中的初始化流程: 3.1 加载配置文件 3.2 扫描相关的类,扫描 ...
- OpenCV+TensorFlow图片手写数字识别(附源码)
初次接触TensorFlow,而手写数字训练识别是其最基本的入门教程,网上关于训练的教程很多,但是模型的测试大多都是官方提供的一些素材,能不能自己随便写一串数字让机器识别出来呢?纸上得来终觉浅,带着这 ...
- javaScript(js)手写原生任务定时器源码
javaScript(js)手写原生任务定时器 功能介绍 定时器顾名思义就是在某个特定的时间去执行一些任务,现代的应用程序早已不是以前的那些由简单的增删改查拼凑而成的程序了,高复杂性早已是标配,而任务 ...
- Spring系列之手写一个SpringMVC
目录 Spring系列之IOC的原理及手动实现 Spring系列之DI的原理及手动实现 Spring系列之AOP的原理及手动实现 Spring系列之手写注解与配置文件的解析 引言 在前面的几个章节中我 ...
- Spring Boot启动命令参数详解及源码分析
使用过Spring Boot,我们都知道通过java -jar可以快速启动Spring Boot项目.同时,也可以通过在执行jar -jar时传递参数来进行配置.本文带大家系统的了解一下Spring ...
- Spring PropertyResolver 占位符解析(二)源码分析
Spring PropertyResolver 占位符解析(二)源码分析 Spring 系列目录(https://www.cnblogs.com/binarylei/p/10198698.html) ...
- 6 手写Java LinkedHashMap 核心源码
概述 LinkedHashMap是Java中常用的数据结构之一,安卓中的LruCache缓存,底层使用的就是LinkedHashMap,LRU(Least Recently Used)算法,即最近最少 ...
- 3 手写Java HashMap核心源码
手写Java HashMap核心源码 上一章手写LinkedList核心源码,本章我们来手写Java HashMap的核心源码. 我们来先了解一下HashMap的原理.HashMap 字面意思 has ...
随机推荐
- 逻辑式编程语言极简实现(使用C#) - 2. 一道逻辑题:谁是凶手
本系列前面的文章: 逻辑式编程语言极简实现(使用C#) - 1. 逻辑式编程语言介绍 这是一道Prolog经典的练习题,中文翻译版来自阮一峰的文章<Prolog 语言入门教程>. 问题 B ...
- Nuxt+Express后端api接口配置与实现方式
Nuxt.js 是一个基于 Vue.js 的轻量级应用框架,可用来创建服务端渲染 (SSR) 应用.本文带你了解在 Nuxt.js 中使用 Express 如何编写实现后端的 api 接口. 创建接口 ...
- 洛谷 P4047 [JSOI2010]部落划分
这道题其实就是无线通讯网的双倍经验啦,只是在输出的时候不同罢了.还是一样的\(kruskal\)算法,但是在求的时候,应该在\(now=n-k+1\)的时候结束.本来到\(n-k\)就行了的,但是由于 ...
- I/O模式及select、 poll、 epoll
I/O多路复用技术 复用技术(multiplexing)并不是新技术而是一种设计思想,在通信和硬件设计中存在频分复用.时分复用.波分复用.码分复用等.在日常生活中复用的场景也非常多.从本质上来说,复用 ...
- mongodb 3.x以上版本与mongodb 2.x版本语法区别
2.x const MongoClient = require('mongodb').MongoClient; const url = 'mongodb://localhost:27017/test' ...
- tomcat发布时候jar包问题
今天遇到个问题就是,启动tomcat时,报:java.lang.NullPointerException at org.apache.jsp.**_jsp.jspInit(index_jsp.java ...
- 美国6w刀的远程工作高级工程师职位,说下在线评估, 倒在第一阶段, 认知能力测试?智商不够怎么办?!
前几天刚被裁员了, 然后在Linkedin上面看到一个crossover的senior software engineer的职位,写的可以remote, 6w刀, 我第一次参加这个公司的这种在线测试, ...
- Redis做为缓存的几个问题
缓存理流程: 前台请求,后台先从缓存中取数据,取到直接返回结果,取不到时从数据库中取,数据库取到更新缓存,并返回结果,数据库也没取到,那直接返回空结果. 1.缓存雪崩 解决方案3:如果缓存数据库是分布 ...
- 【高并发】面试官问我如何使用Nginx实现限流,我如此回答轻松拿到了Offer!
写在前面 最近,有不少读者说看了我的文章后,学到了很多知识,其实我本人听到后是非常开心的,自己写的东西能够为大家带来帮助,确实是一件值得高兴的事情.最近,也有不少小伙伴,看了我的文章后,顺利拿到了大厂 ...
- day44 初识数据库
目录 一.数据的演变 二.数据库 三.MySQL 1 基本原理 2 重要概念介绍 3 安装 4 启动 5 sql基本语句 6 环境变量的配置及系统服务制作 7 关于密码 8 统一编码 9 基本sql语 ...