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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定