Spring IoC 容器原理

Spring 是什么

  • Spring 是 Java 世界应用的事实标准。(事实标准是指非由标准化组织制定的,而是由处于技术领先地位的企业、企业集团制定,由市场实际接纳的技术标准)
  • 没有 Spring 的年代我们怎么开发 Java 应用?
    1.一个 main 程序打天下,这么做规模上来以后难以维护,就是异常灾难。
    2.根据功能拆分为多个模块,拆分并且手动管理所有的依赖,依赖关系纷繁复杂。
  • Spring 容器就是一个 IoC 容器。
  • Spring MVC 基于 Spring 和 Servlet 的 Web 应⽤框架。Servlet 就是一个小服务器,假如我们开发一个 Web 应用,一个进程必须去监听某个端口,世界上所有的人都可以通过这个端口来访问你的程序。我们将这种最底层的,处理交互的东西叫做 Servlet(Java 中大名鼎鼎的 Servlet 容器有 Tomcat 和 Jetty)。Servlet 的功能就是把外界的HTTP请求封装成一个对象交给上层的 WebApp(Java 中 WebApp 的事实标准就是 Spring),WebApp 处理完后交给 Servlet,然后 Servlet 帮你发送出去。所谓的 Spring MVC 就是支持 Servlet 和网络请求的这么一个 Spring 程序,换一句话说,Spring MVC 就是在 Spring 容器之上的一个功能更强大的应用
  • Spring Boot 集成度和⾃动化程度更⾼。(Spring Boot 内嵌了一个 Servlet 容器,操作更加傻瓜了)Spring Boot 是近年来才火起来的,高度集成化。给我们开发人员的直观感受就是,减少了很多配置工作,但是我们丧失了对于底层的所有控制权。
  • 总的来说,Spring 的出现解放了我们的双手。

Spring 容器核心概念

下面我们演示一个最精简的 Spring 程序,模拟一个简单的订单服务。
OrderDao 主要完成一些数据库操作。

public class OrderDao {
    public void select() {
        System.out.println("select from database");
    }
}

OrderSerive 提供对外的服务。

public class OrderService {
    private OrderDao orderDao;

    public void doSomething() {
        System.out.println("do some thing");
    }
}

首先需要引入一个第三方类库 Spring Context(这里对于 context 比较好的解释应该是“环境”,“上下文”总觉得怪怪的)。现在,用的比较多的是 Spring Boot,在上古年代,人们使用 spring xml 来配置 Spring 容器。在 src/main/resource 下新建 config.xml 文件。我们从官网抄来了配置文件,根据自己的需求,添加了两个 bean。现在解释一下 bean 是什么。程序员都是非常浪漫的,Java 是咖啡,而 Bean 是咖啡豆。一个 Bean 可以去依赖其他 Bean,本质上,Bean 就是一个 Java 对象(其实 Spring 整个程序,也是一个 Java 对象,只不过容器给我们的概念是一个很大的盒子)。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="
        http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <!-- bean definitions here -->
    <bean id="orderDao" class="com.github.lazyben.OrderDao"/>
    <bean id="orderService" class="com.github.lazyben.OrderService"/>
</beans>

容器是什么呢?其实就是 BeanFactory。现在我们可以去完成一些操作了。发现了没有整个程序我们没有手工去处理依赖关系。

public class SpringMain {
    public static void main(String[] args) {
        BeanFactory beanFactory = new ClassPathXmlApplicationContext("classpath:config.xml");
        final OrderService orderService = (OrderService) beanFactory.getBean("orderService");
        final OrderDao orderDao = (OrderDao) beanFactory.getBean("orderDao");
        System.out.println(orderService);
        orderService.doSomething();
    }
}

此时报了一个空指针异常。因为此时 orderService 中的 orderDao 是 null。怎么办呢?

com.github.lazyben.OrderService@57f23557
Exception in thread "main" java.lang.NullPointerException
    at com.github.lazyben.OrderService.doSomething(OrderService.java:7)
    at com.github.lazyben.SpringMain.main(SpringMain.java:11)

最简单的就是基于注解的,我们可以添加注解 @Inject(注入)或 @Resource(资源)或 @Autowired(自动装配)。为什么我们添加一个注解就能完成注入依赖的工作呢?这个注解其实本身没什么卵用,只是在运行时被扫描到有这个注解,并完成了工作。这里就不多赘述了,这不是本篇所要讨论的内容。另外,我们还需要在 config.xml 中加入一行配置 <context:annotation-config/>。多说一句,这里比较推荐使用 @Inject 注解,不仅Spring能识别它,Google 的 Guice 也能识别它。
Spring 也支持构造器注入。我们可以把这些注解写在构造器上(称为构造器注入),在属性上写注解有一个很要命的问题,一旦脱离 Spring 就无法工作了,而写在构造器上我们可以在没有 Spring 的情况下通过构造器的方式传入属性(就和写原生 Java 代码一样),写单元测试也更加的方便。构造器注入是一种更好的实践。

public class OrderService {
//    @Inject
//    @Resource
    @Autowired
    private OrderDao orderDao;

    public void doSomething() {
        orderDao.select();
    }
}
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="
           http://www.springframework.org/schema/beans
           http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
           http://www.springframework.org/schema/context
           http://www.springframework.org/schema/context/spring-context-3.1.xsd">

    <context:annotation-config/>
    <!-- bean definitions here -->
    <bean id="orderDao" class="com.github.lazyben.OrderDao"/>
    <bean id="orderService" class="com.github.lazyben.OrderService"/>
</beans>

打上断点,神奇的事情发生了,orderService 中的 orderDao,和 orderDao 居然还是同一个对象。为什么上文说 Spring 解放了我们的双手?道理就在这里。



这里我们回顾一下上文中提到的一些 Spring 的核心概念:

  • Bean
    容器中的最⼩⼯作单元,通常为⼀个 Java 对象。
  • BeanFactory/ApplicationContext
    它们都是容器本身对应的Java对象。
    容器本身对应的 Java 对象。
  • 依赖注⼊ Dependency Injection(DI)
    容器负责注⼊所有的依赖。
  • 控制反转 Inverse of Control(IoC)
    ⽤户将控制权交给了容器。

手写一个最精简的 IoC 容器

无非是以下几个步骤。定义Bean、加载 Bean 的定义、实例化 Bean、查找依赖实现自动注入。

  1. 为了简单,我们使用 .properties 格式文件作为配置文件,在 src/main/resources 下创建配置文件 ioc.properties。propweties 使用等号划分一个键值对。
orderDao=com.github.lazyben.OrderDao
orderService=com.github.lazyben.OrderServicew
  1. 使用一个古老的 Java 类 Properties 可以解析 properties 文件,于是我们拿到了配置文件。
public class MyIoCContainer {
    public static void main(String[] args) throws IOException {
        Properties properties = new Properties();
        properties.load(MyIoCContainer.class.getResourceAsStream("/ioc.properties"));
        System.out.println("properties = " + properties);
    }
}
  1. 根据拿到的配置,使用反射生成实例。
Map<String, Object> beans = new HashMap<>();
properties.forEach((beanName, beanFullyQualifiedClassName) -> {
    try {
        Class<?> klass = Class.forName((String) beanFullyQualifiedClassName);
        Object newInstance = klass.getConstructor().newInstance();
        beans.put((String) beanName, newInstance);
    } catch (Exception e) {
        throw new RuntimeException(e);
    }
});
  1. 依赖注入。对于每一个实例,获取它的所有带 @Autowired 注解的属性,在 beans 中找到相应的实例,并注入依赖。值得注意的是 private 属性,我们不能直接用反射直接设置,需要设置权限 field.setAccessible(true);
beans.forEach((beanName, beanInstance) -> dependencyInject(beanInstance, beans));

private static void dependencyInject(Object beanInstance, Map<String, Object> beans) {
    final List<Field> fieldsToBeAutoWired = Stream.of(beanInstance.getClass().getDeclaredFields())
            .filter(bean -> bean.getAnnotation(Autowired.class) != null)
            .collect(Collectors.toList());

    fieldsToBeAutoWired.forEach(field -> {
        try {
            final Object dependencyBeanInstance = beans.get(field.getName());
            field.setAccessible(true);
            field.set(beanInstance, dependencyBeanInstance);
        } catch (IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    });
 }
  1. 源码
public class MyIoCContainer {
    public static void main(String[] args) throws IOException {
        Properties properties = new Properties();
        properties.load(MyIoCContainer.class.getResourceAsStream("/ioc.properties"));

        Map<String, Object> beans = new HashMap<>();
        properties.forEach((beanName, beanFullyQualifiedClassName) -> {
            try {
                Class<?> klass = Class.forName((String) beanFullyQualifiedClassName);
                Object newInstance = klass.getConstructor().newInstance();
                beans.put((String) beanName, newInstance);
            } catch (Exception e) {
                throw new RuntimeException(e);
            }
        });

        beans.forEach((beanName, beanInstance) -> dependencyInject(beanInstance, beans));

        final OrderService orderService = (OrderService) beans.get("orderService");
        orderService.doSomething();
    }

    private static void dependencyInject(Object beanInstance, Map<String, Object> beans) {
        final List<Field> fieldsToBeAutoWired = Stream.of(beanInstance.getClass().getDeclaredFields())
                .filter(bean -> bean.getAnnotation(Autowired.class) != null)
                .collect(Collectors.toList());

        fieldsToBeAutoWired.forEach(field -> {
            try {
                final Object dependencyBeanInstance = beans.get(field.getName());
                field.setAccessible(true);
                field.set(beanInstance, dependencyBeanInstance);
            } catch (IllegalAccessException e) {
                throw new RuntimeException(e);
            }
        });
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容