Summer——从头开始写一个简易的Spring框架

              

​ 参考Spring框架实现一个简易类似的Java框架。计划陆续实现IOCAOP、以及数据访问模块和事务控制模块。项目持续维护中...欢迎Star!Thanks~~~

​ 本项目参考自Spring,但又做出了自己的创新,最大的创新点在于解决循环依赖的问题,引入了自己的解决方案,具体方法可以参考更新日志中Version 0.5(Pre-release)的更新记录。

项目计划:

  • [x] IOC容器
  • [x] AOP切面
  • [x] 对外的扩展接口
  • [ ] 数据访问集成模块(JDBC、事务控制)

​ 关于对IOC和AOP功能为什么要使用(why),以及应该如何使用(how)请移步使用文档,要了解每个版本更新的内容请移步更新日志

​ 关于如何将自己的框架适配到summer,请查看如何适配summer

运行环境

JDK 8

项目主要技术栈

注解、反射

如何使用

​ 仓库地址: https://github.com/vfdxvffd/Summer

​ 下载最新的jar包 ,将其导入项目中,即可使用,目录结构如下图,蓝色框内为summer的核心代码,ch包下为logback日志依赖,net.sf.cglib下为cglib动态代理的依赖,org.slf4j下为slf4j的日志门面依赖。

Version 1.0(Release)

本次更新测试了之前的版本的稳定性,以及增加了框架的扩展性,开放出别的框架适配的接口。

  • 对之前的pre-release版本进行了多次测试以确保稳定性。
  • 框架对外开放Extension接口,该接口中的方法在ioc容器构造的多个阶段进行了切入,增加了框架的可扩展性。
  • 如何将自己的框架适配到Summer,请看如何适配summer或者可以提issues

Version 0.5(Pre-release)

一次重大更新

bug描述:循环依赖的问题复现出来

​ 因为之前v0.1更新中引入的一个解决bug的方法导致了这个重大的bug,这次通过设置二级缓存来解决循环依赖的问题,具体bug的产生原因详情可见更新日志,更新日志对这次bug的出现原因以及解决方法做了详细的说明。

bug解决:

​ 针对目前掌握的代理方面的知识,对之前的做法做出一些调整。设置二级缓存,一级缓存一个(即真正的ioc容器),二级缓存两个,都是负责存放实例化但未初始化的对象,但一个是存放原对象,另一个负责存放代理对象,二级缓存的示意图如下:

将ioc容器的构造过程分为四步来进行:

  1. 遍历包,找到所有需要被IOC管理的类,封装成BeanDefinition
  2. 根据第一步获取到的BeanDefinition实例化那些单例且非延迟加载的对象,并将其加入到二级缓存的earlyRealObjects
  3. 对第二步得到的earlyRealObjects中的对象进行检查,看是否需要设置代理,如果需要则对其进行代理,并将代理对象加入到二级缓存中的earlyProxyObjects中(并不删除earlyRealObjects中对应的真正的对象)。
  4. 对第二步得到的earlyRealObjects中的对象进行注入工作(即开始进行初始化),检查每个对象的每个域,如果标注了@Autowired注解且值为null,则对其进行注入工作,现在一级缓存中查找,如果有直接取出为其注入,如果没有检查二级缓存的earlyProxyObjects,如果有则取出为其注入,如果没有则接着检查二级缓存的earlyRealObjects,找到后为其注入,此时如果还没有则说明这个域对应的bean是非单例(prototype)模式或者懒加载模式的,则为其实例化并设置代理(如果需要),并初始化,然后注入其中。如果是非ioc容器管理的域,则直接注入null,也可以考虑改为抛出异常给用户提示。

Version 0.4(Pre-release)

本次更新加入了新功能,修改了一个已知的bug

  • 本次更新引入CGLib依赖,增加动态代理的方式,对于实现了接口的方法采用JDK动态代理来实现切面功能,对于没有实现接口的类采用CGLib来实现切面。

  • 修改bug,之前版本中的判断当前类是否已经完成了实例对象全部的创建注入工作的方法,判断没有包含所有情况。

    bug描述:对于一个没有任何域需要代理的对象,进行注入工作的时候会由于没有域需要注入,从而直接判断其已经完成注入,而跳过了代理阶段。

Version 0.3(Pre-release)

  • 本次更新引入了日志依赖,增加了对ioc构造过程中的日志记录

  • 对于标注了@Aspect注解的类自动将其加入IOC容器中,不用再重复标注注解

Version 0.2(Pre-release)

本次更新加入了一些新功能,修复了一些bug

  1. 更新功能:

    • aop增加了一种切入方式,目前有以下切入方式

      @Before@AfterReturning@AfterThrowing@After

      以上对应的切入时机如下:

      try {
      @Before
      fun.invoke();
      @AfterReturning
      } catch (Throwable t) {
      @AfterThrowing
      } finally {
      @After
      }
    • 切面方法可以通过JoinPoint类获取被切的方法的参数、方法名、返回值类型。对于@AfterReturning的切入方式可以获取返回值,类型为Object,而@AfterThrowing可以获取抛出的异常,类型为Throwable

  2. 修复了重复代理的bug

    bug描述:当一个待注入bean中有超过一个需要注入的域(带有注解@Autowired且未完成赋值),如果对它中的方法进行切面,这时切面方法会重复执行

Version 0.1(Pre-release)

​ 本次更新主要修复了一些bug,以及优化了代码的结构

  1. 修复对于注入对象的切面方法失效的bug

    bug描述:在controller中注入service,但是如果有对于service的切面方法,则切面方法无法被调用

  2. 修复延迟加载的对象注入失败的bug

    bug描述:对于标注了延迟加载的类注入时会发生异常

  3. 修复对非单例的bean注入失败的bug

    bug描述:对于标注了非单例的类注入时会发生异常,且会调用多次构造函数的问题

  4. 增加核心代码的注释。

  5. 优化代码结构,重构了大部分冗余的代码块

  6. 抽取可重用方法。

Version 0.0(Pre-release)

  1. 完成IOC容器的初步搭建

  2. 完成AOP功能的简单使用(还需修改)

  3. 支持@Component@Autowired@Qualifier@Value@Repository@Service@Controller注解的使用

    • @Component(同@Respository、@Service、@Controller):标注在类上,将此类注册到ioc容器中
    • @Autowired:自动注入ioc容器中的对象
    • @Qualifier:自动注入ioc中对象的时候指定beanName,如不指定则按照beanType注入
    • @Value:指定将类注入到容器是基本类型(包括包装类)字段的值
  4. 支持根据beanNamebeanType获取ioc中的对象

  5. 自定义类型转化异常,@Value接受String类型,如果传入的值并不能正确转化,就抛出DataConversionException异常。

  6. 增加单例模式与非单例模式的配置注解@Scope,以及增加延迟加载的配置注解@Lazy

  7. 可以使用接口来接受IOC中返回的对象

  8. AOP可以对方法进行@Before@After@AfterThrowing的切面,需要配置方法的全方法名

  9. AOP使用JDK的动态代理,set可以不添加,内部实现是直接通过设置域的可访问属性,然后直接设置值

  10. 后续计划:

    • 支持根据xml配置ioc容器中的对象
    • 对于运行过程可能发生的异常使其尽可能可控,且明确的抛出或处理
    • 对于AOP可选择性的加入CGLIB代理
    • 对于AOP一些已注入对象的代理失效bug进行修复(已定位)
    • etc... for more...

Summer——从头开始写一个简易的Spring框架的更多相关文章

  1. 造轮子:实现一个简易的 Spring IoC 容器

    作者:DeppWang.原文地址 我通过实现一个简易的 Spring IoC 容器,算是入门了 Spring 框架.本文是对实现过程的一个总结提炼,需要配合源码阅读,源码地址. 结合本文和源码,你应该 ...

  2. 用Python写一个简单的Web框架

    一.概述 二.从demo_app开始 三.WSGI中的application 四.区分URL 五.重构 1.正则匹配URL 2.DRY 3.抽象出框架 六.参考 一.概述 在Python中,WSGI( ...

  3. go server框架学习之路 - 写一个自己的go框架

    go server框架学习之路 - 写一个自己的go框架 用简单的代码实现一个go框架 代码地址: https://github.com/cw731/gcw 1 创建一个简单的框架 代码 packag ...

  4. 动手写一个简单的Web框架(模板渲染)

    动手写一个简单的Web框架(模板渲染) 在百度上搜索jinja2,显示的大部分内容都是jinja2的渲染语法,这个不是Web框架需要做的事,最终,居然在Werkzeug的官方文档里找到模板渲染的代码. ...

  5. 动手写一个简单的Web框架(Werkzeug路由问题)

    动手写一个简单的Web框架(Werkzeug路由问题) 继承上一篇博客,实现了HelloWorld,但是这并不是一个Web框架,只是自己手写的一个程序,别人是无法通过自己定义路由和返回文本,来使用的, ...

  6. 动手写一个简单的Web框架(HelloWorld的实现)

    动手写一个简单的Web框架(HelloWorld的实现) 关于python的wsgi问题可以看这篇博客 我就不具体阐述了,简单来说,wsgi标准需要我们提供一个可以被调用的python程序,可以实函数 ...

  7. 手写一个简易的IOC

    这个小项目是我读过一点Spring的源码后,模仿Spring的IOC写的一个简易的IOC,当然Spring的在天上,我写的在马里亚纳海沟,哈哈 感兴趣的小伙伴可以去我的github拉取代码看着玩 地址 ...

  8. 来,我们手写一个简易版的mock.js吧(模拟fetch && Ajax请求)

    预期的mock的使用方式 首先我们从使用的角度出发,思考编码过程 M1. 通过配置文件配置url和response M2. 自动检测环境为开发环境时启动Mock.js M3. mock代码能直接覆盖g ...

  9. 如何用 Python 写一个简易的抽奖程序

    不知道有多少人是被这个头图骗进来的:) 事情的起因是这样的,上周有同学问小编,看着小编的示例代码敲代码,感觉自己也会写了,如果不看的话,七七八八可能也写的出来,但是一旦自己独立写一段程序,感觉到无从下 ...

随机推荐

  1. 根据数据渲染DOM树形菜单——中途感想

    根据数据渲染DOM树形菜单,这个需求做了几天了.一开始觉得用while也可以实现一层一层查找数据,但后来发现while还是做不到,因为我查找这个动作必须有进入有出来然后进入下一个条目,但while只能 ...

  2. 攻防世界 reverse serial-150

    serial-150 suctf-2016 直接使用ida发现main函数中夹杂大片数据,应该是自修改代码,动态调试: 调试中发现,输入为16位,验证方法为:从头开始取一字符进行比较,比较通过检验后, ...

  3. 5、Spring教程之依赖注入

    概念 依赖注入(Dependency Injection,DI). 依赖 : 指Bean对象的创建依赖于容器 . Bean对象的依赖资源 . 注入 : 指Bean对象所依赖的资源 , 由容器来设置和装 ...

  4. MySQL Order BY 排序过程

    MySQL 在进行 Order By 操作排序时,通常有两种排序方式: 全字段排序 Row_id 排序 MySQL 中每个线程在执行排序时,都会被分配一块区域 - sort buffer,它的大小通过 ...

  5. Webpack 5 配置手册(从0开始)

    针对新手入门搭建项目,Webpack5 配置手册(从0开始) webpack安装顺序 1. `npm init -y`,初始化包管理文件 package.json 2. 新建src源代码目录 3. 新 ...

  6. 开源框架TLog核心原理架构解析

    前言 最近在做TLog 1.2.5版本的迭代,许多小伙伴之前也表示说很想参与开源项目的贡献.为了让项目更好更快速的迭代新特性以及本着发扬开源精神互相学习交流,很有幸招募到了很多小伙伴与我一起前行. 为 ...

  7. [Azure Devops] 使用 Azure Repos 管理代码

    1. 什么是 Azure Repos Azure Repos 是一组版本控制工具,可用于管理代码.无论您的软件项目是大型项目还是小型项目,都应尽快使用版本控制. 版本控制系统是可帮助您跟踪随时间变化对 ...

  8. 【linux】驱动-9-设备树插件

    目录 前言 9. Linux设备树插件 9.1 格式 9.2 设备树插件的编译和加载 9.2.1 单独使用dtc工具编译 9.2.2 内核dtc工具编译设备树插件 9.2.3 加载设备树插件 9.2. ...

  9. 多线程安全的单例模式(使用判断nullptr和call_once两种方法)

    转载请注明: https://blog.csdn.net/Stephen___Qin/article/details/115583694 使用判断nullptr #include <thread ...

  10. ASP.NET Core中使用令牌桶限流

    在限流时一般会限制每秒或每分钟的请求数,简单点一般会采用计数器算法,这种算法实现相对简单,也很高效,但是无法应对瞬时的突发流量. 比如限流每秒100次请求,绝大多数的时间里都不会超过这个数,但是偶尔某 ...