Spring IOC 三大高级特性(简述)

一、lazy-init 延迟加载

延迟加载主要是指Bean对象的延迟创建。ApplicationContext 容器的默认行为是在容器启动时将所有的single bean 提前实例化。提前实例化意味着初始化过程的一部分,ApplicationContext 实力会创建并配置所有的singleton bean。

属性:

  • lazy-init=“false”,false表示立即加载,true表示延迟加载,默认是false。立即加载表示spring启动时立即加载,延迟加载则在ApplicationContext 启动时不会立即实例化,而是在第一次向容器getBean索取bean时才会实例化。
  • default-lazy-init=“false”,在容器层次中通过在元素上使用"default-lazy-init" 属性来控制延时初始化。

使用:

<bean id="testBean" calss="com.LazyBean" lazy-init="true" />
<beans default-lazy-init="true">
	<!-- no beans will be eagerly pre-instantiated... -->
</beans>

注:如果⼀个 bean 的 scope 属性为 scope=“pototype” (多例)时,即使设置了 lazy-init=“false”,容器启动时也不会实例化bean,⽽是调⽤ getBean ⽅法实例化的。

应用场景:

  • 使用延迟加载一定程度上提高容器启动和运转性能;
  • 对于不常使用的bean设置延迟加载,这样不必从一开始就占用资源。

二、FactoryBean 和 BeanFactory

BeanFactory:接口是容器的顶级接口,定义了容器的一些基础行为,是负责生产和管理bean的一个工厂,具体使用它下面的子接口类型,例如:ApplicationContext。

FactoryBean:
spring中的bean分为两种,一种是普通bean,一种是工厂bean(FactoryBean),FactoryBean可以生成某个类型的bean实例返回给我们,因此我们可以借助它自定义bean的创建过程。

源代码:

// 可以让我们⾃定义Bean的创建过程(完成复杂Bean的定义)
public interface FactoryBean<T> {
	@Nullable
	// 返回FactoryBean创建的Bean实例,如果isSingleton返回true,则该实例会放到Spring容器的单例对象缓存池中Map
	T getObject() throws Exception;
	@Nullable
	// 返回FactoryBean创建的Bean类型
	Class<?> getObjectType();

	// 返回作⽤域是否单例
	default boolean isSingleton() { 
		return true;
	}
}

使用实例
实体:

@Data
public class Company {
	private String name;
	private String address;
	private int scale;
}

实现FactoryBean接口:

public class CompanyFactoryBean implements FactoryBean<Company> {
	private String companyInfo; // 公司名称,地址,规模
	
	public void setCompanyInfo(String companyInfo) {
		this.companyInfo = companyInfo;
	}
	
	@Override
	public Company getObject() throws Exception {
		// 模拟创建复杂对象Company
		Company company = new Company();
		String[] strings = companyInfo.split(",");
		company.setName(strings[0]);
		company.setAddress(strings[1]);
		company.setScale(Integer.parseInt(strings[2]));
		return company ;
	}

	@Override
	public Class<?> getObjectType() { 
		return Company.class;
	}

	@Override
	public boolean isSingleton() { 
		return true;
	}
}

xml配置:

<bean id="companyBean" class="com.edu.factory.CompanyFactoryBean">
	<property name="companyInfo" value="xx,中关村,500"/>
</bean>

测试:

Object companyBean = applicationContext.getBean("companyBean");
System.out.println("bean:" + companyBean);

//获取FanctoryBean
Object companyBean = applicationContext.getBean("&companyBean"); System.out.println("bean:" + companyBean);

结果如下:

bean:Company{name='xx', address='中关村', scale=500}

bean:com.edu.factory.CompanyFactoryBean@53f6fd09

三、后置处理器

Spring提供了两种后处理bean的扩展接⼝,分别为 BeanPostProcessor 和BeanFactoryPostProcessor。

BeanFactoryPostProcessor:
在BeanFactory初始化之后可以使⽤BeanFactoryPostProcessor进⾏后置处理做⼀些事情。BeanFactory级别的处理,是针对整个Bean的⼯⼚进⾏处理。

典型应⽤:PropertyPlaceholderConfigurer
在这里插入图片描述
此接⼝只提供了⼀个⽅法,⽅法参数为 ConfigurableListableBeanFactory ,该参数类型定义了⼀些⽅法,其中有个⽅法名为 getBeanDefinition 的⽅法,我们可以根据此⽅法,找到我们定义bean的 BeanDefinition 对象,然后我们可以对定义的属性进⾏修改,⽅法名字类似我们bean标签的属性, setBeanClassName 对应bean标签中的class属性,以下是 BeanDefinition 中的⽅法:
在这里插入图片描述
BeanDefinition对象:我们在 XML 中定义的 bean标签,Spring 解析 bean 标签成为⼀个 JavaBean,这个JavaBean 就是 BeanDefinition。

注意:调⽤ BeanFactoryPostProcessor ⽅法时,这时候bean还没有实例化,此时 bean 刚被解析成BeanDefinition对象。

BeanPostProcessor:在Bean对象实例化(并不是Bean的整个⽣命周期完成)之后可以使⽤BeanPostProcessor进⾏后置处理做⼀些事情。是针对Bean级别的处理,可以针对某个具体的Bean。
在这里插入图片描述
该接⼝提供了两个⽅法,分别在 Bean 的初始化⽅法前和初始化⽅法后执⾏,这个初始化⽅法类似我们在定义bean时,定义了 init-method 所指定的⽅法。

使用:
定义⼀个类实现了 BeanPostProcessor,默认是会对整个Spring容器中所有的bean进⾏处理。如果要对具体的某个bean处理,可以通过⽅法参数判断,两个类型参数分别为Object和String,第⼀个参数是每个bean的实例,第⼆个参数是每个bean的name或者id属性的值。所以我们可以通过第⼆个参数,来判断我们将要处理的具体的bean。

注:处理是发⽣在Spring容器的实例化和依赖注⼊之后。

springBean的生命周期

图如下:
在这里插入图片描述
Bean 生命周期的整个执行过程描述如下。

1)根据配置情况调用 Bean 构造方法或工厂方法实例化 Bean。

2)利用依赖注入完成 Bean 中所有属性值的配置注入。

3)如果 Bean 实现了 BeanNameAware 接口,则 Spring 调用 Bean 的 setBeanName() 方法传入当前 Bean 的 id 值。

4)如果 Bean 实现了 BeanFactoryAware 接口,则 Spring 调用 setBeanFactory() 方法传入当前工厂实例的引用。

5)如果 Bean 实现了 ApplicationContextAware 接口,则 Spring 调用 setApplicationContext() 方法传入当前 ApplicationContext 实例的引用。

6)如果 BeanPostProcessor 和 Bean 关联,则 Spring 将调用该接口的预初始化方法 postProcessBeforeInitialzation() 对 Bean 进行加工操作,此处非常重要,Spring 的 AOP 就是利用它实现的。

7)如果 Bean 实现了 InitializingBean 接口,则 Spring 将调用 afterPropertiesSet() 方法。

8)如果在配置文件中通过 init-method 属性指定了初始化方法,则调用该初始化方法。

9)如果 BeanPostProcessor 和 Bean 关联,则 Spring 将调用该接口的初始化方法 postProcessAfterInitialization()。此时,Bean 已经可以被应用系统使用了。

10)如果在 < bean > 中指定了该 Bean 的作用范围为 scope=“singleton”,则将该 Bean 放入 Spring IoC 的缓存池中,将触发 Spring 对该 Bean 的生命周期管理;如果在 < bean > 中指定了该 Bean 的作用范围为 scope=“prototype”,则将该 Bean 交给调用者,调用者管理该 Bean 的生命周期,Spring 不再管理该 Bean。

11)如果 Bean 实现了 DisposableBean 接口,则 Spring 会调用 destory() 方法将 Spring 中的 Bean 销毁;如果在配置文件中通过 destory-method 属性指定了 Bean 的销毁方法,则 Spring 将调用该方法对 Bean 进行销毁。

Spring 为 Bean 提供了细致全面的生命周期过程,通过实现特定的接口或 < bean > 的属性设置,都可以对 Bean 的生命周期过程产生影响。虽然可以随意配置 < bean > 的属性,但是建议不要过多地使用 Bean 实现接口,因为这样会导致代码和 Spring 的聚合过于紧密。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Spring IoC 的实现过程主要分为两个步骤:首先是通过配置文件或注解将需要管理的对象注册到 Spring 容器中,然后在需要使用这些对象的时候,通过容器来获取这些对象的实例。具体来说,Spring IoC 的实现过程包括以下几个步骤: 1. 配置文件或注解:在 Spring 中,我们可以通过 XML 配置文件或注解的方式来将需要管理的对象注册到 Spring 容器中。在配置文件中,我们可以使用 <bean> 标签来定义一个 Bean,指定其 ID 和 Class,以及其他的属性和依赖关系。在注解中,我们可以使用 @Component、@Service、@Controller 等注解来标记一个类,表示这个类是一个 Bean。 2. Bean 的实例化:当 Spring 容器启动时,会根据配置文件或注解中的信息,创建所有需要管理的 Bean 的实例。这个过程中,Spring 会根据配置文件或注解中的信息,使用反射机制来创建 Bean 的实例,并且自动解决 Bean 之间的依赖关系。 3. Bean 的装配:在 Bean 实例化之后,Spring 会根据配置文件或注解中的信息,将 Bean 之间的依赖关系进行装配。这个过程中,Spring 会根据配置文件或注解中的信息,自动将一个 Bean 中需要依赖的其他 Bean 注入到这个 Bean 中。 4. Bean 的生命周期:在 Spring 容器启动时,会创建所有需要管理的 Bean 的实例,并且自动解决 Bean 之间的依赖关系。在 Bean 实例化之后,Spring 会调用 Bean 的初始化方法,进行一些初始化操作。在 Spring 容器关闭时,会调用 Bean 的销毁方法,进行一些清理操作。 总之,Spring IoC 的实现过程主要是通过配置文件或注解将需要管理的对象注册到 Spring 容器中,并且在需要使用这些对象的时候,通过容器来获取这些对象的实例。 ### 回答2: Spring IoC(Inversion of Control,控制反转)是Spring框架的核心特性之一,它通过管理和控制应用中的对象依赖关系来实现。 Spring IoC的实现过程如下: 1. 根据应用的配置文件编写相应的Bean定义。Spring IoC容器通过读取配置文件,了解应用中需要管理的bean的信息,包括bean的类名、作用域、依赖关系等。 2. 创建和管理Bean实例。Spring IoC容器根据配置文件中的定义,实例化需要管理的Bean,并根据其作用域进行管理。例如,对于单例作用域的Bean,容器只会创建一个实例,并在整个应用中共享。 3. 处理Bean的依赖关系。Spring IoC容器会自动解析Bean之间的依赖关系,确保每个Bean都能获取到它所依赖的其他Bean。容器根据配置文件中的定义,进行依赖注入。常见的注入方式有构造函数注入、setter方法注入和字段注入。 4. 提供Bean的生命周期管理。Spring IoC容器负责管理Bean的生命周期,包括初始化和销毁。容器在实例化Bean之后,会调用其生命周期回调方法进行初始化操作;在容器关闭时,会调用Bean的销毁方法进行资源释放等操作。 5. 提供AOP(面向切面编程)支持。Spring IoC容器还提供了AOP的支持,通过将横切关注点(如日志记录、事务管理等)分离出来,使得系统的业务逻辑和其他关注点可以独立演化。 综上所述,Spring IoC通过配置文件和容器的管理,实现了对象的创建、依赖注入和生命周期管理,从而实现了对象之间的解耦和灵活性。通过使用Spring IoC,我们可以更方便地管理和控制应用中的对象,提高了代码的可维护性和可测试性。 ### 回答3: Spring IoC(Inversion of Control)是一种通过反转对象创建和管理的控制方式,它将对象的创建和依赖注入的过程交由IoC容器来管理。下面是Spring IoC的实现过程: 1. 配置文件定义:首先,我们需要在Spring配置文件或者使用注解方式(@ComponentScan、@Configuration等)来定义Bean的配置。配置文件包括Bean的名字、类路径和其他属性。 2. 加载配置文件:Spring IoC容器负责加载配置文件,并将其解析成内部数据结构。它读取配置文件中的内容,包括Bean的定义、依赖关系和其他属性。 3. 创建Bean实例:Spring IoC容器根据配置文件中定义的Bean的信息,创建Bean的实例。它使用Java反射机制来实例化对象,并提供了灵活的方式来自定义Bean的创建过程。可以通过构造方法、工厂方法或者使用第三方库进行Bean的创建。 4. 设置Bean属性:一旦Bean的实例创建完成,Spring IoC容器将会为Bean设置其对应的属性。这些属性可以通过Setter和Getter方法进行设置和获取。Spring IoC容器根据配置文件中定义的依赖关系,自动为Bean注入它所依赖的其他Bean。 5. Bean的生命周期管理:Spring IoC容器管理Bean的整个生命周期。它在Bean的实例创建完成之后,会调用初始化方法来完成一些初始化操作。同时,当Bean不再被使用时,容器会调用销毁方法来进行资源释放。 6. 提供Bean的引用:Spring IoC容器负责将Bean的引用提供给其他组件使用。其他组件可以通过调用容器提供的方法来获取Bean的实例,从而完成对Bean的操作和调用。 总结来说,Spring IoC实现的过程包括了配置文件的定义、加载和解析、Bean的实例化和属性设置、Bean的生命周期管理以及Bean的引用提供。通过这种方式,Spring IoC实现了对象的解耦和依赖的管理,提高了代码的灵活性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值