查看原文
其他

从零开始造Spring

刘欣 码农翻身 2018-10-26
事实证明我是有拖延症的,而且还不轻。 比如去年搞了两季编程提高群,第一季是JVM , 第二季是面向对象的设计, 本来计划去年12月初开第三季造轮子(Spring),没想到一下子从去年拖到了现在!


有人说,你定的目标知道的人越多,这个目标完成的可能性越大。于是痛下决心,年初的时候在知识星球的几千人面前挖了几个坑,列了一下上半年的Todo, 其中有一个就是 编程提高群第三季的造轮子(Spring)课程。


这么多双眼睛盯着,坑不能不填,不填就下不了台了。


为什么是Spring?


大家都说“不要重复发明轮子”, 那是没有意义的,确实是这样。 但是我们可以再造一遍, 在造的过程中来学习别人优秀的设计和优雅的代码。


那为什么要造Spring的轮子?


很简单, Spring 几乎就是Java 后端开发的代名词,统治了十几年,人家还在与时俱进,如果没有意外, 估计还要继续统治下去,这样的软件和系统,绝对值得Java 后端程序员投资,深入学习。


到底怎么造?


这个问题一度让我非常纠结。


我已经把Spring 核心的IoC和AOP看了一遍,说实话,很痛苦。 难道带着大家去通读源码,然后再仿照着写?


可是Spring实在是太庞大了,仅仅是核心IoC和AOP代码,即使是早期版本,依然是一篇黑暗森林: 每个阅读源码的程序员都像一个带枪的猎人,小心翼翼地在丛林中穿行,一不小心就......

如果这么做,大家很有可能陷入Spring细节的枝枝蔓蔓中, 会从入门到放弃。


还有一种办法,那就是写个极简版本,几百行代码就把最最核心的东西实现了。这样的代码网上也不少, 但是似乎和学习Spring的目的不符啊!


我们学习Spring到底学习什么?


我想主要是看人家是怎么设计的,对于一些概念是怎么抽象的,代码是怎么组织的...... 基于这个前提,带着大家再去读源码是一件低效率的事情,不能这么做!


后来我想到了一个办法: 从零开始,一步步逼近Spring 。


从最简单的结构(也就是基于XML的BeanFactory,只有缺省构造函数的Bean)开始,一步步扩展到setter注入, 有参数的构造函数,注解,auto-scan , 用cglib实现AOP,用java动态代理实现AOP......


在这个过程中,不能天马行空地写代码,脑海中始终有一根弦:一定要时刻注意着和Spring 的核心思想,核心类保持一致,具体的做法就是:不断地重构


这样以来,我们最终的代码会和Spring的核心类非常类似,只是去掉了一些处理各种各样异常情况的枝枝蔓蔓, Spring的主干会得以保留,即逼近Spring,而不是Copy Spring。


在实现的过程中,你会看到Spring为什么是这么设计的,为什么要做出Resource, BeanDefinitionRegistry,InjectedElement,BeanPostProcessor......这样的抽象。 换句话说,我们要理解why, 而不仅仅是How。  我觉得这也是造Spring轮子的最大价值所在。


特别值得一提的是,我会采用测试驱动(TDD)的方案,完全站在使用者的角度,用测试先行的方式驱动出所有的代码。对单元测试和重构不熟悉的同学,可以看看是怎么完成TDD这个过程的,TDD是如何和设计结合的。


总之,利用这种从零开始,一步步逼近Spring的办法,相当于重现了Spring框架的诞生过程(当然具体的思路和原作者不一定相同)既使得造轮子过程不那么枯燥,又能理解Spring源码的核心思想。


我希望生成的代码“简约而不简单”,既能体现Spring的设计思想,又不会过分复杂。


为什么没有SpringMVC?


我发现,仅仅是逼近Spring 的IoC和AOP的核心功能,就已经非常复杂了。


考虑到SpringMVC更为复杂,尤其是和Web相关的,很多细节的处理代码,这次还是放弃。


课程提纲


第一周:概述

课程简介

介绍Spring IoC, AOP

介绍TDD开发方式, 重构的方法。

第二周:Basic BeanFactory

最简单的结构,基于XML的BeanFactory

缺省构造函数的Bean

Resource,BeanDefinitionRegistry 的抽象

Excpeiton的处理

BeanDefinition接口

单一职责的应用

第三周:实现setter注入

Bean的scope问题,SingletonBeanRegistry接口

PropertyValue,RuntimeBeanReference,BeanDefinitionValueResolver的抽象

使用Introspector

TypeConverter 实现从字符串到特定类型的转换

从createBean到initiateBean和populateBean的拆分。

第四周: 实现构造函数注入

引入ConstructorArgument

如何找到合适的构造器: ConstructorResolver

TestCase的整理,引入TestSuite

第五、六周:实现注解和auto-scan

注解的讲解

使用ASM读取类的Metadata

读取一个包下所有的class 作为Resource 

新的BeanDefinition实现类引入

给auto-scan的Bean 命名: BeanNameGenerator

新的抽象: DependencyDescriptor,InjectionMetadata ,InjectedElement

用AutowiredAnnotationProcessor实现注入

Bean的生命周期

BeanPostProcessor接口及其实现和调用

第七、八、九周:实现AOP

准备工作:讲解CGLib和Java动态代理的原理,讲解PointCut, Advice....等AOP概念

实现Pointcut 和 MethodMatcher,MethodLocatingFactory

给定一个对象及相应的方法和一系列拦截器(beforeAdvice,afterAdvice),需要实现正确的调用次序

实现CGLibProxyFactory

实现“合成”Bean

实现对resolveInnerBean

使用AspectJAutoProxyCreator 进行封装

用Java 动态代理实现AOP

课程总结。


课程形式和时间


这次课程主要面向使用过Spring,并且对Spring 内部实现和源码感兴趣的同学。


由于代码量很大,直播缓慢的敲代码过程意义不大, 这次会采用录像+答疑(直播)的方式。


6月10号开始,每周日发布一次视频,每周三QQ群直播答疑,当然,QQ群里随时交流。


费用


想立刻参与的话,可以给我微信或者支付宝转账,199元


微信:onlyliuxin97

支付宝 : onlyliuxin@gmail.com


转账后务必加我的微信 onlyliuxin97 或者QQ 3471198853, 我会邀请你入群。

    您可能也对以下帖子感兴趣

    文章有问题?点此查看未经处理的缓存