广告位联系
返回顶部
分享到

萌芽版的Spring容器的实现方法

java 来源:互联网 作者:秩名 发布时间:2022-03-04 19:21:35 人浏览
摘要

从什么是IOC开始? Spring春天,Java编程世界的春天是由一位音乐家Rod Johnson带来的。 Rod Johnson先后编写了两本巨著《Expert One-on-One J2EE Design and Development》、《Expert One-on-One J2EE Development

从什么是IOC开始?

Spring——春天,Java编程世界的春天是由一位音乐家——Rod Johnson带来的。

Rod Johnson先后编写了两本巨著《Expert One-on-One J2EE Design and Development》、《Expert One-on-One J2EE Development without EJB》,拉起了挑战正统Java EE框架EJB的大旗。

Rod Johnson两大著作-来自百度百科

Rod Johnson不仅是一名旗手,更是开发了Spring这一轻量级框架,像一名勇敢的龙骑兵一样,对EJB发动了冲锋,并最终战胜了EJB,让Spring成为Java EE事实上的标准。

Spring Logo

Spring的两大内核分别是IOC和AOP,其中最最核心的是IOC。

所谓的IOC(控制反转):就是由容器来负责控制对象的生命周期和对象间的关系。以前是我们想要什么,就自己创建什么,现在是我们需要什么,容器就给我们送来什么。

引入IOC之前和引入IOC之后

也就是说,控制对象生命周期的不再是引用它的对象,而是容器。对具体对象,以前是它控制其它对象,现在所有对象都被容器控制,所以这就叫控制反转。

控制反转示意图

也许你还听到另外一个概念DI(依赖注入),它指的是容器在实例化对象的时候把它依赖的类注入给它,我们也可以认为,DI是IOC的补充和实现。

工厂和Spring容器

Spring是一个成熟的框架,为了满足扩展性、实现各种功能,所以它的实现如同枝节交错的大树一样,现在让我们把视线从Spring本身移开,来看看一个萌芽版的Spring容器怎么实现。

Spring的IOC本质就是一个大工厂,我们想想一个工厂是怎么运行的呢?

工厂运行

生产产品:一个工厂最核心的功能就是生产产品。在Spring里,不用Bean自己来实例化,而是交给Spring,应该怎么实现呢?——答案毫无疑问,反射。

那么这个厂子的生产管理是怎么做的?你应该也知道——工厂模式。

库存产品:工厂一般都是有库房的,用来库存产品,毕竟生产的产品不能立马就拉走。Spring我们都知道是一个容器,这个容器里存的就是对象,不能每次来取对象,都得现场来反射创建对象,得把创建出的对象存起来。

订单处理:还有最重要的一点,工厂根据什么来提供产品呢?订单。这些订单可能五花八门,有线上签签的、有到工厂签的、还有工厂销售上门签的……最后经过处理,指导工厂的出货。

在Spring里,也有这样的订单,它就是我们bean的定义和依赖关系,可以是xml形式,也可以是我们最熟悉的注解形式。

那对应我们的萌芽版的Spring容器是什么样的呢?

mini版本Spring IOC

订单:Bean定义

Bean可以通过一个配置文件定义,我们会把它解析成一个类型。

Bean定义

  • beans.properties

为了偷懒,这里直接用了最方便解析的properties,用一个<key,value>类型的配置来代表Bean的定义,其中key是beanName,value是class

userDao:cn.fighter3.bean.UserDao

  • BeanDefinition.java

bean定义类,配置文件中bean定义对应的实体

1

2

3

4

5

6

7

public class BeanDefinition {

 

    private String beanName;

 

    private Class beanClass;

     //省略getter、setter 

 }

获取订单:资源加载

接下订单之后,就要由销售向生产部门交接,让生产部门知道商品的规格、数量之类。

资源加载器,就是来完成这个工作的,由它来完成配置文件中配置的加载。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

public class ResourceLoader {

 

    public static Map<String, BeanDefinition> getResource() {

        Map<String, BeanDefinition> beanDefinitionMap = new HashMap<>(16);

        Properties properties = new Properties();

        try {

            InputStream inputStream = ResourceLoader.class.getResourceAsStream("/beans.properties");

            properties.load(inputStream);

            Iterator<String> it = properties.stringPropertyNames().iterator();

            while (it.hasNext()) {

                String key = it.next();

                String className = properties.getProperty(key);

                BeanDefinition beanDefinition = new BeanDefinition();

                beanDefinition.setBeanName(key);

                Class clazz = Class.forName(className);

                beanDefinition.setBeanClass(clazz);

                beanDefinitionMap.put(key, beanDefinition);

            }

            inputStream.close();

        } catch (IOException | ClassNotFoundException e) {

            e.printStackTrace();

        }

        return beanDefinitionMap;

    }

 

}

订单分配:Bean注册

对象注册器,这里用于单例bean的缓存,我们大幅简化,默认所有bean都是单例的。可以看到所谓单例注册,也很简单,不过是往HashMap里存对象。

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

public class BeanRegister {

 

    //单例Bean缓存

    private Map<String, Object> singletonMap = new HashMap<>(32);

 

    /**

     * 获取单例Bean

     *

     * @param beanName bean名称

     * @return

     */

    public Object getSingletonBean(String beanName) {

        return singletonMap.get(beanName);

    }

 

    /**

     * 注册单例bean

     *

     * @param beanName

     * @param bean

     */

    public void registerSingletonBean(String beanName, Object bean) {

        if (singletonMap.containsKey(beanName)) {

            return;

        }

        singletonMap.put(beanName, bean);

    }

 

}

生产车间:对象工厂

好了,到了我们最关键的生产部门了,在工厂里,生产产品的是车间,在IOC容器里,生产对象的是BeanFactory。

BeanFactory

对象工厂,我们最核心的一个类,在它初始化的时候,创建了bean注册器,完成了资源的加载。

获取bean的时候,先从单例缓存中取,如果没有取到,就创建并注册一个bean

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

47

public class BeanFactory {

 

    private Map<String, BeanDefinition> beanDefinitionMap = new HashMap<>();

 

    private BeanRegister beanRegister;

 

    public BeanFactory() {

        //创建bean注册器

        beanRegister = new BeanRegister();

        //加载资源

        this.beanDefinitionMap = new ResourceLoader().getResource();

    }

 

    /**

     * 获取bean

     *

     * @param beanName bean名称

     * @return

     */

    public Object getBean(String beanName) {

        //从bean缓存中取

        Object bean = beanRegister.getSingletonBean(beanName);

        if (bean != null) {

            return bean;

        }

        //根据bean定义,创建bean

        return createBean(beanDefinitionMap.get(beanName));

    }

 

    /**

     * 创建Bean

     *

     * @param beanDefinition bean定义

     * @return

     */

    private Object createBean(BeanDefinition beanDefinition) {

        try {

            Object bean = beanDefinition.getBeanClass().newInstance();

            //缓存bean

            beanRegister.registerSingletonBean(beanDefinition.getBeanName(), bean);

            return bean;

        } catch (InstantiationException | IllegalAccessException e) {

            e.printStackTrace();

        }

        return null;

    }

}

生产销售:测试

UserDao.java

我们的Bean类,很简单

1

2

3

4

5

6

public class UserDao {

 

    public void queryUserInfo(){

        System.out.println("A good man.");

    }

}

单元测试

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

public class ApiTest {

    @Test

    public void test_BeanFactory() {

        //1.创建bean工厂(同时完成了加载资源、创建注册单例bean注册器的操作)

        BeanFactory beanFactory = new BeanFactory();

 

        //2.第一次获取bean(通过反射创建bean,缓存bean)

        UserDao userDao1 = (UserDao) beanFactory.getBean("userDao");

        userDao1.queryUserInfo();

 

        //3.第二次获取bean(从缓存中获取bean)

        UserDao userDao2 = (UserDao) beanFactory.getBean("userDao");

        userDao2.queryUserInfo();

    }

}

运行结果

A good man.
A good man.

至此,我们一个萌芽版的Spring容器就完成了。

考虑一下,它有哪些不足呢?是否还可以抽象、扩展、解耦……

细细想想这些东西,你是不是对真正的Spring IOC容器为何如此复杂,有所理解了呢?

参考:

[1].《Spring揭秘》

[2].小傅哥 《手撸Spring》

[3].《精通Spring4.X企业应用开发实战》


版权声明 : 本文内容来源于互联网或用户自行发布贡献,该文观点仅代表原作者本人。本站仅提供信息存储空间服务和不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权, 违法违规的内容, 请发送邮件至2530232025#qq.cn(#换@)举报,一经查实,本站将立刻删除。
原文链接 : https://blog.csdn.net/sinat_40770656/article/details/123217592
相关文章
  • SpringBoot自定义错误处理逻辑介绍

    SpringBoot自定义错误处理逻辑介绍
    1. 自定义错误页面 将自定义错误页面放在 templates 的 error 文件夹下,SpringBoot 精确匹配错误信息,使用 4xx.html 或者 5xx.html 页面可以打印错误
  • Java实现手写一个线程池的代码

    Java实现手写一个线程池的代码
    线程池技术想必大家都不陌生把,相信在平时的工作中没有少用,而且这也是面试频率非常高的一个知识点,那么大家知道它的实现原理和
  • Java实现断点续传功能的代码

    Java实现断点续传功能的代码
    题目实现:网络资源的断点续传功能。 二、解题思路 获取要下载的资源网址 显示网络资源的大小 上次读取到的字节位置以及未读取的字节
  • 你可知HashMap为什么是线程不安全的
    HashMap 的线程不安全 HashMap 的线程不安全主要体现在下面两个方面 在 jdk 1.7 中,当并发执行扩容操作时会造成环形链和数据丢失的情况 在
  • ArrayList的动态扩容机制的介绍

    ArrayList的动态扩容机制的介绍
    对于 ArrayList 的动态扩容机制想必大家都听说过,之前的文章中也谈到过,不过由于时间久远,早已忘却。 所以利用这篇文章做做笔记,加
  • JVM基础之字节码的增强技术介绍

    JVM基础之字节码的增强技术介绍
    字节码增强技术 在上文中,着重介绍了字节码的结构,这为我们了解字节码增强技术的实现打下了基础。字节码增强技术就是一类对现有字
  • Java中的字节码增强技术

    Java中的字节码增强技术
    1.字节码增强技术 字节码增强技术就是一类对现有字节码进行修改或者动态生成全新字节码文件的技术。 参考地址 2.常见技术 技术分类 类
  • Redis BloomFilter布隆过滤器原理与实现

    Redis BloomFilter布隆过滤器原理与实现
    Bloom Filter 概念 布隆过滤器(英语:Bloom Filter)是1970年由一个叫布隆的小伙子提出的。它实际上是一个很长的二进制向量和一系列随机映射
  • Java C++算法题解leetcode801使序列递增的最小交换次

    Java C++算法题解leetcode801使序列递增的最小交换次
    题目要求 思路:状态机DP 实现一:状态机 Java 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 class Solution { public int minSwap(int[] nums1, int[] nums2) { int n
  • Mybatis结果集映射与生命周期介绍

    Mybatis结果集映射与生命周期介绍
    一、ResultMap结果集映射 1、设计思想 对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了 2、resultMap的应用场
  • 本站所有内容来源于互联网或用户自行发布,本站仅提供信息存储空间服务,不拥有版权,不承担法律责任。如有侵犯您的权益,请您联系站长处理!
  • Copyright © 2017-2022 F11.CN All Rights Reserved. F11站长开发者网 版权所有 | 苏ICP备2022031554号-1 | 51LA统计