spring-core-3-26 | Spring IoC容器:BeanFactory和ApplicationContext谁才是Spring IoC容器?

2024-02-16 09:18

本文主要是介绍spring-core-3-26 | Spring IoC容器:BeanFactory和ApplicationContext谁才是Spring IoC容器?,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Spring IoC 容器

BeanFactory 和ApplicationContext 谁才是Spring IoC 容器?

真正的IOC的底层实现就是BeanFactory的实现类,但ApplicationContext 在底层组合了一个 BeanFactory 实现,是委托DefaultListableBeanFactory来操作getBean等方法的。

也因此
System.out.println(userRepository.getBeanFactory() ==
applicationContext); 是false尽管他们都复用了同样的接口 BeanFactory, 但他们不是一个对象

代码示例

/** Licensed to the Apache Software Foundation (ASF) under one or more* contributor license agreements.  See the NOTICE file distributed with* this work for additional information regarding copyright ownership.* The ASF licenses this file to You under the Apache License, Version 2.0* (the "License"); you may not use this file except in compliance with* the License.  You may obtain a copy of the License at**     http://www.apache.org/licenses/LICENSE-2.0** Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/
package org.geekbang.thinking.in.spring.ioc.overview.dependency.injection;import org.geekbang.thinking.in.spring.ioc.overview.repository.UserRepository;
import org.springframework.beans.factory.BeanFactory;
import org.springframework.beans.factory.ObjectFactory;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;/**** 26 | Spring IoC容器:BeanFactory和ApplicationContext谁才是Spring IoC容器?* 见下面的 whoIsIoCContainer()** @author <a href="mailto:mercyblitz@gmail.com">Mercy</a>* @since*/
public class DependencyInjectionDemo {/*** 26 | Spring IoC容器:BeanFactory和ApplicationContext谁才是Spring IoC容器?** @param userRepository* @param applicationContext*/private static void whoIsIoCContainer(UserRepository userRepository,//BeanFactory beanFactoryApplicationContext applicationContext) {/*ApplicationContext其实就是BeanFactory, 为什么这么说?首先要参看spring官方文档对于beanFactory和applicationContext的解释:https://docs.spring.io/spring-framework/docs/5.2.2.RELEASE/spring-framework-reference/core.html1.1节中The org.springframework.beans and org.springframework.context packages are the basisfor Spring Framework’s IoC container.org.springframework.beans 和 org.springframework.context是 spring Ioc容器的相关包The BeanFactory interface provides an advanced configuration mechanism capable ofmanaging any type of object.BeanFactory接口提供了一些高级配置的一个机制, 能够来管理任何类型的对象.(这里用object, 而不是bean, 是非常准确的, 因为我们前面知道, IoC的依赖来源有bean也有依赖(非bean))ApplicationContext is a sub-interface of BeanFactory. It adds:ApplicationContext 是 BeanFactory的子接口, 同时还提供了下面的特性:Easier integration with Spring’s AOP features简化了和spring AOP特性的整合Message resource handling (for use in internationalization)消息资源的处理(用于国际化)Event publication事件的发布, 就是spring的事件Application-layer specific contexts such as the WebApplicationContext foruse in web applications.用于应用级别的上下文, 比如WebApplicationContext 使用在web场景的应用下In short, the BeanFactory provides the configuration framework and basic functionality,总而言之, BeanFactory提供了一个配置的框架且是基本的功能and the ApplicationContext adds more enterprise-specific functionality.并且 ApplicationContext 增加了一些企业级应用的特性.到这里我们可以认为BeanFactory就是一个基本的IoC容器, 而ApplicationContext是它的一个超集The ApplicationContext is a complete superset of the BeanFactory and is used exclusivelyin this chapter in descriptions of Spring’s IoC container.For more information on using the BeanFactory instead of the ApplicationContext, see [beans-beanfactory].*/// 这个表达式为什么不会成立System.out.println(userRepository.getBeanFactory() == applicationContext);/*ApplicationContext is BeanFactory, 如官方文档所说, ApplicationContext就是BeanFactory的一个超集我们从子类往上找, 按照这样的方向去找找名堂:ApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:/META-INF/dependency-injection-context.xml");ClassPathXmlApplicationContext => AbstractXmlApplicationContext => AbstractRefreshableConfigApplicationContext=> AbstractRefreshableApplicationContext => AbstractApplicationContext在最后的抽象类 AbstractApplicationContext 中是这样的public abstract class AbstractApplicationContext extends DefaultResourceLoaderimplements ConfigurableApplicationContext {ConfigurableApplicationContext 是配置模式, 这是一种可写的模式在这个类中寻找方法 getBeanFactory()@Overridepublic abstract ConfigurableListableBeanFactory getBeanFactory() throws IllegalStateException;可见也是覆盖了接口的方法, 那么这个接口就是 ConfigurableApplicationContext那么这里就有些奇怪, 看类头是这样的:public interface ConfigurableApplicationContext extends ApplicationContext, Lifecycle, Closeable {ApplicationContext不就是BeanFactory的子接口么ConfigurableApplicationContext <- ApplicationContext <- BeanFactory这里有两个问题:为什么ConfigurableApplicationContext要特意提供一个方法getBeanFactory()? 直接获取不到父类对象么?为什么System.out.println(userRepository.getBeanFactory() == applicationContext); 是false呢?那么ConfigurableApplicationContext既然是可写模式, 就回提供set方法:可以找到void setParent(@Nullable ApplicationContext parent);但是没有 setBeanFactory??往回找到 ConfigurableApplicationContext 的实现类  AbstractApplicationContext 的 getBeanFactory()这里就给按层往子类找, 有可能是在中间的抽象类实现的,用 ctrl + F12 搜索方法, 会发现实现是放在中间的抽象类 AbstractRefreshableApplicationContext@Overridepublic final ConfigurableListableBeanFactory getBeanFactory() {synchronized (this.beanFactoryMonitor) {if (this.beanFactory == null) {throw new IllegalStateException("BeanFactory not initialized or already closed - " +"call 'refresh' before accessing beans via the ApplicationContext");}return this.beanFactory;}}看一下类中beanFactory变量:@Nullableprivate DefaultListableBeanFactory beanFactory;会发现这个 beanFactory 是一个组合的, 就是说这里代码是将 beanFactory 的实现 DefaultListableBeanFactory来组合进来了, 并不是去完全抽象或继承的这个类.那么这也就解释了上面的两个问题userRepository.getBeanFactory()获取的也正是 DefaultListableBeanFactory这里先提供一个先入为主的概念:在上下文里面的实现是采用了一种组合的方式, 同时在接口上面又是 extends 关系这种方式有点像代理再去看getBean()的实现:在 AbstractApplicationContext 中:@Overridepublic <T> T getBean(Class<T> requiredType) throws BeansException {assertBeanFactoryActive();return getBeanFactory().getBean(requiredType);}这里就是先getBeanFactory(), 相当于我们先用代理对象去查找BeanFactory的实现, 并不是自己要具备这样的能力,而是外面组装了一个组合对象来帮助我们做这个事情结论是真正的IOC的底层实现就是BeanFactory的实现类,但 ApplicationContext 在底层组合了一个 BeanFactory 实现,是委托DefaultListableBeanFactory来操作getBean等方法的。也因此 System.out.println(userRepository.getBeanFactory() == applicationContext); 是false尽管他们都复用了同样的接口 BeanFactory, 但他们不是一个对象*/}}

这篇关于spring-core-3-26 | Spring IoC容器:BeanFactory和ApplicationContext谁才是Spring IoC容器?的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/714116

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2