【框架源码】手写Spring框架IOC容器核心流程

这篇具有很好参考价值的文章主要介绍了【框架源码】手写Spring框架IOC容器核心流程。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【框架源码】手写Spring框架IOC容器核心流程

要是想要了解Spring IOC底层,首先我们就得先了解什么是IOC。

IOC就是控制反转,把对象创建和对象之间的调用过程交给Spring进行管理。

使用IOC目的就是之前我们创建对象的方式都是用new的方式创建,这个方式有个缺点,被创建对象的类的位置一旦被改动,new就会报错,类与类之间的关联度太大,为了降低耦合度,将创建对象交给IOC容器处理。

IOC底层原理也是运用主要技术包括:xml解析、工厂设计模式、反射。

IOC主要是通过内部工厂类解析配置文件id属性对应的class属性,利用反射Class.forName()方法,参数是class属性值,来回去对应的类,进而获取对象。

下面我们就来用自己的方式来实现Spring IOC的核心功能点。

Spring IOC 核心功能点如下:

  • 支持读取xml文件,创建多个bean
  • 提供接口根据id获取容器bean
  • 根据xml配置,需要支持对象注入功能
  • 支持判断容器里面是否存在bean和获取class类型
  • 采用适当的接口抽取和封装,使用方式和spring框架基本类似

这是我们在Spring框架中创建Bean的方式:

【框架源码】手写Spring框架IOC容器核心流程

接下来,我们就按照Spring的这中创建Bean的方式自定义一个简易的IOC框架。

创建一个maven项目。

【框架源码】手写Spring框架IOC容器核心流程

首先,我们先加入一个依赖,然后创建一些环境测试类,OrderMapper、OrderService、OrderServiceImpl、beans.xml。

		<!--JDOM 是一种使用 XML 的独特 Java 工具包,用于快速开发 XML 应用程序。JDOM 是一个开源项目,它基于树形结构,利用纯 Java 的技术对 XML 文档实现解析、生成、序列化及多种操作。-->
		<dependencies>
        <dependency>
            <groupId>org.jdom</groupId>
            <artifactId>jdom</artifactId>
            <version>2.0.2</version>
        </dependency>
    </dependencies>
public class OrderMapper {

    public void saveOrder(String orderNo){
        System.out.println("当前订单:"+orderNo+" 已经落入数据库。");
    }

}
public interface OrderService {

    void saveOrder(String orderNo);
}
public class OrderServiceImpl implements OrderService {

  	//注意这个OrderMapper 的 set、get方法一定要写,我们一会用反射的机制去设置属性。
    private OrderMapper orderMapper;

    public OrderMapper getOrderMapper() {
        return orderMapper;
    }

    public void setOrderMapper(OrderMapper orderMapper) {
        this.orderMapper = orderMapper;
    }

    @Override
    public void saveOrder(String orderNo) {
        orderMapper.saveOrder(orderNo);
    }
}
<?xml version="1.0" encoding="UTF-8" ?>
<beans>
    <bean id="orderMapper" class="com.lixiang.mapper.OrderMapper">
    </bean>

    <bean id="orderService" class="com.lixiang.service.impl.OrderServiceImpl">
        <property name="orderMapper" bean="orderMapper" />
    </bean>
</beans>

ok,准备完测试类,我们来定义BeanFactory,仿照Spring的方式,定义几个方法。

/**
 * @author lixiang
 * @date 2023/5/22 08:26
 */
public interface BeanFactory {

    /**
     * 根据名称获取bean
     * @param name
     * @return
     */
    Object getBean(String name);

    /**
     * 判断bean是否在容器里面
     * @param name
     * @return
     */
    boolean containsBean(String name);

    /**
     * 根据名称获取bean的类型
     * @param name
     * @return
     */
    Class<?> getType(String name) ;

}

然后创建ClassPathXmlApplicationContext,上下文对象,用于加载resources下的xml文件。

/**
 * @author lixiang
 * @date 2023/5/22 08:29
 */
public class ClassPathXmlApplicationContext implements BeanFactory{


    /**
     * 定义map用于存放bean
     */
    private Map<String,Object> beans = new HashMap<>();

    public ClassPathXmlApplicationContext() throws Exception {
        //init 方法 做 解析和注入
        init();
    }

    /**
     * 初始化,核心就在这个
     */
    private void init() throws Exception {

        SAXBuilder saxBuilder = new SAXBuilder();
        //获取资源路径的xml文件
        InputStream resourceAsStream = this.getClass().getClassLoader().getResourceAsStream(fileName);
        //将xml文件构建成Document
        Document document = saxBuilder.build(resourceAsStream);
        //获取根结点的Element,对应的就是beans
        Element rootElement = document.getRootElement();
        //获取下面的bean节点
        List<Element> elements = rootElement.getChildren("bean");
        //遍历
        for (int i = 0; i < elements.size(); i++) {
            Element element = elements.get(i);
            //获取id属性
            String id = element.getAttributeValue("id");
            //获取class信息
            String clazz = element.getAttributeValue("class");
            //反射创建对象
            Object obj = Class.forName(clazz).getDeclaredConstructor().newInstance();
            //放到map中存储
            beans.put(id, obj);
            //判断属性中是否有需要注入的对象
            for(Element propertyElement: element.getChildren("property")){
                //获取名称
                String name = propertyElement.getAttributeValue("name");
                //获取bean的定义
                String bean = propertyElement.getAttributeValue("bean");
                //获取bean
                Object beanObject = beans.get(bean);
                //设置set方法
                String methodName = "set" + name.substring(0,1).toUpperCase() + name.substring(1);
                Class<?> aClass = beanObject.getClass();
                //反射将属性设置进去
                Method m = obj.getClass().getMethod(methodName,aClass);
                m.invoke(obj, beanObject);
            }
        }
    }

    @Override
    public Object getBean(String name) {
        return beans.get(name);
    }

    @Override
    public boolean containsBean(String name) {
        return beans.containsKey(name);
    }

    @Override
    public Class<?> getType(String name) {
        if(beans.containsKey(name)){
            return beans.get(name).getClass();
        }
        return null;
    }
}

创建测试主函数,按照之前Spring 创建的方式写。

/**
 * @author lixiang
 * @date 2023/5/22 08:11
 */
public class Main {
    public static void main(String[] args) throws Exception {

        String fileName= "beans.xml";
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext(fileName);

        OrderService orderService = (OrderService)applicationContext.getBean("orderService");
        orderService.saveOrder("202388474754758834");
        boolean containsBean = applicationContext.containsBean("orderService");
        System.out.println(containsBean);
        Class<?> type = applicationContext.getType("orderService");
        System.out.println(type.getName());

    }
}

我们先看运行结果,没有问题,对象被创建,并且调用方法。

【框架源码】手写Spring框架IOC容器核心流程

那么下面我们看debug看下它是怎么执行的,核心逻辑就在于init方法。

【框架源码】手写Spring框架IOC容器核心流程
【框架源码】手写Spring框架IOC容器核心流程
【框架源码】手写Spring框架IOC容器核心流程
【框架源码】手写Spring框架IOC容器核心流程
Spring中顶层类 BeanFactory

  • 最核心的接口之一,主要负责创建和管理bean对象实例,具体包括定义、加载、实例化和维护Bean之间的依赖关系等
  • 主要作用
    • 加载Bean的配置信息:BeanFactory根据XML文件中定义的Bean信息构造Bean实例,并装载到容器中。
    • 实例化Bean:BeanFactory在Bean的定义信息加载后,利用Java反射机制来实例化Bean,并根据依赖关系装配Bean实例。
    • 维护Bean之间的依赖关系:BeanFactory能够自动识别Bean之间的依赖关系,实现Bean的依赖注入
    • 提供统一的配置方式:BeanFactory可以将所有Bean的配置信息放在一起,提供统一的配置入口。
    • 对Bean进行作用域管理:BeanFactory负责对Bean的作用域进行管理,如:单例、多例等。

BeanFactory 只是个接口,并不是IOC容器的具体实现,Spring容器给出了很多种实现,

  • ClassPathXmlApplicationContext 使用XML配置
  • AnnotationConfigApplicationContext 使用注解配置
  • ApplicationContextConfigurableApplicationContext
  • BeanFactoryListableBeanFactory
  • 总结
    • BeanFactory是顶层接口,定义多数最基础的API,称为【基础容器】
    • 对应BeanFactory的子类ApplicationContext,可以基于不同需求拓展更多的功能,称为【高级容器】
    • 这样的设计避免全部功能都集中在一个类,分散到不同接口,实现的时候根据需求选择即可

【框架源码】手写Spring框架IOC容器核心流程文章来源地址https://www.toymoban.com/news/detail-462013.html

到了这里,关于【框架源码】手写Spring框架IOC容器核心流程的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Spring Framework的核心:IoC容器的实现(1)

    个人名片: 🐼 作者简介:一名大二在校生 🐻‍❄️ 个人主页:落798. 🐼 个人WeChat:落798. 🕊️ 系列专栏: 零基础学java ----- 重识c语言 ---- 计算机网络 — 【Spring技术内幕】 🐓 每日一句: 看淡一点在努力,你吃的苦会铺成你要的路! Spring 容器是 Spring 框架的核心。 容

    2024年02月15日
    浏览(28)
  • Spring框架IOC容器和DI依赖注入

    IOC(Invertion Of Control):控制反转,使用对象时,由使用new创建对象转变为由外部提供对象,此过程中对象的创建控制权由程序转移到外部的思想称之为控制反转. DI(Dependency Injection):依赖注入,在容器中建立bean与bean之间的关系的过程,称之为依赖注入 pom.xml依赖引入 BookDao BookDaoImpl Book

    2023年04月09日
    浏览(73)
  • 深入分析Spring的IoC容器:从底层源码探索

    前言: 博主在最近的几次面试中,大中小厂都问到了Spring的ioc容器相关问题,这块知识确实是面试中的重点内容,因此结合所看的书籍,在这篇文章中总结下。该专栏比较适合刚入坑Java的小白以及准备秋招的大佬阅读,感谢大佬的关注。 如果文章有什么需要改进的地方欢迎

    2024年02月12日
    浏览(40)
  • spring 容器结构/机制debug分析--Spring 学习的核心内容和几个重要概念--IOC 的开发模式--综合解图

    😀前言 本片文章是spring 容器结构/机制debug分析和几个重要概念以及IOC 的开发模式 🏠个人主页:[尘觉主页](https://blog.csdn.net/apple_67445472?type=blog) 🧑个人简介:大家好,我是尘觉,希望我的文章可以帮助到大家,您的满意是我的动力😉😉 在csdn获奖荣誉: 🏆csdn城市之星2名

    2024年02月02日
    浏览(49)
  • 【Spring源码解读一】IoC容器之AnnotationConfigApplicationContext

    AnnotationConfigApplicationContext构造方法 this() AnnotatedBeanDefinitionReader ClassPathBeanDefinitionScanner register(componentClasses)         根据AnnotationConfigApplicationContext类去阅读其将Bean对象交给IoC容器管理的过程。以下这三个代码块是将配置类注册进IoC容器的例子。下面是关于这个类的继承与

    2024年02月08日
    浏览(34)
  • 【spring源码系列-02】通过refresh方法剖析IOC的整体流程

    Spring源码系列整体栏目 内容 链接地址 【一】spring源码整体概述 https://blog.csdn.net/zhenghuishengq/article/details/130940885 【二】通过refresh方法剖析IOC的整体流程 https://blog.csdn.net/zhenghuishengq/article/details/131003428 【三】xml配置文件启动spring时refresh的前置工作 https://blog.csdn.net/zhenghuishen

    2024年02月08日
    浏览(33)
  • 4.是人就能学会的Spring源码教程-IOC容器创建Bean对象

    我们要关注一个接口 BeanFactory ,它是Spring IOC容器的根接口,也是容器的入口。 类的描述中已经清楚的说明了: 我们来看一下这个接口里面的方法。 我们可以看到有各种各样的 getBean 方法,让我们可以从容器中获取到各种各样的Bean对象。 BeanFactory 有一个实现类 DefaultListab

    2024年02月05日
    浏览(28)
  • 【框架源码】Spring源码解析之Bean创建源码流程

    问题:Spring中是如何初始化单例bean的? 我们都知道Spring解析xml文件描述成BeanDefinition,解析BeanDefinition最后创建Bean将Bean放入单例池中,那么Spring在创建Bean的这个过程都做了什么。 Spring核心方法refresh()中最最重要的一个方法 finishBeanFactoryInitialization() 方法,该方法负责初始化

    2024年02月09日
    浏览(28)
  • 手写Springboot核心流程

    目录 Springboot启动流程 核心代码 验证效果 创建Spring容器, 扫描并启动 容器选择Tomcat/Jetty 创建DispatchServlet, 与spring容器绑定 将DispatchServlet添加到Tomcat 启动Tomcat 1. 首先, 创建两个module  2. maven依赖 springboot模块依赖 user模块依赖 3. springboot核心配置类  4. Import自动配置类  spi注入

    2024年02月01日
    浏览(29)
  • 【Spring 核心 | IoC】

      IoC即控制反转(Inversion of Control,缩写为 IoC)。IoC又称为依赖倒置原则(设计模式六大原则之一)。 IoC意味着将你设计好的对象交给容器控制,而不是传统的在你的对象内部直接控制。这样对象的控制权就反转了,你无需关心对象的各种创建,只需关注业务的本身,大大

    2024年02月10日
    浏览(48)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包