手写一个简易的IOC

简介: 这个小项目是我读过一点Spring的源码后,模仿Spring的IOC写的一个简易的IOC,当然Spring的在天上,我写的在马里亚纳海沟,哈哈

这个小项目是我读过一点Spring的源码后,模仿Spring的IOC写的一个简易的IOC,当然Spring的在天上,我写的在马里亚纳海沟,哈哈


感兴趣的小伙伴可以去我的github拉取代码看着玩


地址: https://github.com/zhuchangwu/CIOC

点击查看源码


项目中有两种方式实现IOC:

  • 第一种是基于dom4j实现的解析XML配置文件版
  • 第二种是基于自定义注解实现全配置版


全注解版#


模仿Spring原生的IOC机制如下:

  • Interface类型的beanDefinition不会被实例化**
  • String类型的beanDefinition不会被实例化
  • 维护三个核心的map容器
  • 使用底层存放实例化对象的容器是一个叫singletonObjects的CurrentHashMap
  • 第二个用来辅助解决循环依赖的容器叫singletonFactories类型:CurrentHashMap
  • 第三个用来存放bean定义信息的map容器叫beanDefinitionMap类型:CurrentHashMap


Spring底层的自己还封装了BeanDefinition, 当然我没干这件事,直接用的类的描述对象

Class


自定义了四种注解如下:#


  • CDao 用于标识持久层的对象


@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface CDao {
    String value()default "";
}


  • CService 用来标识服务层的对象


@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface CService {
    String value()default "";
}


  • CComponentScan 用来标识主配置类,提供包扫描需要的base-packet


@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE})
public @interface CComponentScan {
    String value()default "";
}


  • CAutowired 用来标识需要自动装配的对象


@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface CAutowired {
    String value()default "";
}


当然他跟Spring原生的@Autowired是没法比的,Spring自动装配类型默认是Autowired_no, 但是被Spring原生标记上的对象会先按照默认的装配类型进行装配,如果没有默认的装配类型,再按照byType,如果容器中存在多个相同类型的对象,就按照byName, 名字再一样就直接报错了


另外我说一下,说了这么多byName和byType, 那么byName,是通过哪个name呢? 如果是基于setter方法的话就是setter去掉set后,后面的名字, 如果是通过@Autowired的byName,这个name其实就是属性名

Spring是允许程序员去改这个默认的装配类型的


然后在我的IOC中就比较逊色了,直接默认按照byType,没有合适的类型再按照byName进行自动装配


解决了循环依赖的问题#


在我手动写如何解决循环依赖的时候,那时候我还没有去看源码, 当时我画了几个流程图,但是还是卡壳了, 于是我去调试Spring的实现, 简直了!Spring的作者们简直是真神!

其实说Spring如何解决循环依赖的,我前面有几个源码阅读的博客,感兴趣可以去看看

这里我就简单的说下, 这件事是一个叫AutowiredAnnotationBeanDefinitonPostprocessor的后置处理器完成的, Spring在做这件事是时候,前前后后是一个偌大的继承体系在支持,但是归根结底是Spring玩了个漂亮的递归,方法名是getBean(),当然这个递归还有几个辅助容器,这几个容器就是我上面说的几个map ,我的IOC能写成,就得益于这一点


XML版#


注解版的IOC我是用DOM4j解析XML配置文件实现的, 做了下面的功能

  • 支持setter方法依赖注入

标识性的信息是 property


<bean id="dao1" class="com.changwu.dao.DaoImpl1"></bean>
<bean id="service" class="com.changwu.service.UserServiceImpl4">
    <property ref="dao1" name="daoImpl"></property>
</bean>


  • 支持构造方法的依赖注入

标识性的信息是 constructor-arg


<bean id="DaoImpl" class="com.changwu.dao.DaoImpl1"></bean>
<bean id="service" class="com.changwu.service.UserServiceImpl3">
    <constructor-arg ref ="DaoImpl" name="DaoImpl1"></constructor-arg>
</bean>
</bean>


  • 支持byType的自动装配

标识性的信息是 byType


<beans  default-autowire="byType">


  • 主持byName的自动装配


<beans  default-autowire="byName">


感兴趣的小伙伴可以去我的github拉取代码看着玩

地址: https://github.com/zhuchangwu/CIOC

点击查看源码

相关文章
|
8月前
|
SQL XML Java
源码分析系列教程(08) - 手写MyBatis(注解版)
源码分析系列教程(08) - 手写MyBatis(注解版)
61 0
|
18天前
|
XML Java 数据格式
手写一个简易的IOC
手写一个简易的IOC
|
6天前
|
Java API Spring
Spring6(七):手写IoC
Spring6(七):手写IoC
8 0
|
2月前
|
XML Java 数据格式
手写spring 框架——第二篇(依赖注入)
手写spring 框架——第二篇(依赖注入)
47 0
|
2月前
|
缓存 Java Maven
理解IOC 手写springIOC
理解IOC 手写springIOC
28 0
|
2月前
|
Java API Spring
Spring之手写IoC
【1月更文挑战第15天】 我们都知道,Spring框架的IOC是基于Java反射机制实现的,下面我们先回顾一下java反射。
75 3
|
11月前
|
设计模式 Java Spring
用300行代码手写1个Spring框架,麻雀虽小五脏俱全
为了解析方便,我们用application.properties来代替application.xml文件,具体配置内容如下:
30 0
|
11月前
|
监控 前端开发 Java
30个类手写Spring核心原理之AOP代码织入(5)
前面我们已经完成了Spring IoC、DI、MVC三大核心模块的功能,并保证了功能可用。接下来要完成Spring的另一个核心模块—AOP,这也是最难的部分。
42 0
|
11月前
|
缓存 Java Spring
30个类手写Spring核心原理之依赖注入功能(3)
在之前的源码分析中我们已经了解到,依赖注入(DI)的入口是getBean()方法,前面的IoC手写部分基本流程已通。先在GPApplicationContext中定义好IoC容器,然后将GPBeanWrapper对象保存到Map中。在GPApplicationContext中设计两个Map:factoryBeanObjectCache保存单例对象的缓存,factoryBeanInstanceCache保存GPBeanWrapper的缓存,变量命名也和原生Spring一致,这两个对象的设计其实就是注册式单例模式的经典应用。
34 0
|
12月前
|
存储 缓存 Java
纯手写IOC以及解决简单的循环依赖的问题
纯手写IOC以及解决简单的循环依赖的问题