当前位置:首页 > 编程笔记 > 正文
已解决

Spring底层原理(五)

来自网友在路上 170870提问 提问时间:2023-11-04 09:14:18阅读次数: 70

最佳答案 问答题库708位专家为你答疑解惑

Spring底层原理(五)

本章内容

介绍Aware接口与InitializingBean接口、Bean的初始化与销毁、Scope

Aware接口

作用:用于注入一些与容器相关的信息

类名作用BeanNameAware注入Bean的名称BeanFactoryAware注入BeanFactory容器ApplicationContextAware注入ApplicationContext容器EmbeddedValueResolverAware${}

🔖 注入功能使用@Autowired就能实现,为什么还要用Aware接口呢?

  • @Autowired的解析需要用到bean后处理器,属于扩展行为
  • Aware接口属于内置功能,不加任何扩展,Spring就能识别

某些情况下,扩展功能会失效,而内置功能不会失效

InitializingBean接口

该接口提供了一种内置的初始化手段,内置的注入和初始化不受拓展功能的影响,总会被执行,因此Spring框架内部的类常用它

@Autowired失效分析

Java配置类不包含BeanFactoryPostProcessor的情况

在这里插入图片描述

Java配置类包含BeanFactoryPostProcessor的情况,创建其中的BeanFactoryPostProcessor必须提前创建Java配置类,而此时的BeanPostProcessor还未准备好,导致@Autowired等注解失效

解决办法:不要使用@Autowired@PostConstruct注解,而是使用InitializingBean接口

案例代码

@Slf4j
@Configuration
public class Config2 {@Autowiredpublic void app(ApplicationContext context) {log.info("===== 注入容器:{}", context);}@PostConstructpublic void init(){log.info("===== 初始化");}@Beanpublic BeanFactoryPostProcessor processor1(){return beanFactory->{log.info("执行processor1");};}
}
public class A05Application {public static void main(String[] args) throws IOException {GenericApplicationContext context = new GenericApplicationContext();context.registerBean(AutowiredAnnotationBeanPostProcessor.class);context.registerBean(CommonAnnotationBeanPostProcessor.class);context.registerBean(ConfigurationClassPostProcessor.class);context.registerBean("config2",Config2.class);context.refresh();for (String name : context.getBeanDefinitionNames()) {System.out.println(name);}context.close();}
}

更正后

@Slf4j
@Configuration
public class Config2 implements InitializingBean, ApplicationContextAware {@Beanpublic BeanFactoryPostProcessor processor1(){return beanFactory->{log.info("执行processor1");};}@Overridepublic void afterPropertiesSet() throws Exception {log.info("===== 初始化");}@Overridepublic void setApplicationContext(ApplicationContext applicationContext) throws BeansException {log.info("===== 注入容器:{}", applicationContext);}
}
Bean的初始化与销毁
初始化
  • 使用@Bean(initMethod="")
  • 使用@PostConstruct注解
  • 使用InitializingBean接口

执行顺序:@PostConstruct @Bean(initMethod="") InitializingBean

销毁
  • 使用@Bean(distoryMethod="")
  • 使用@PreDistory注解
  • 使用DisposableBean接口

执行顺序:@PreDistory @Bean(distoryMethod="") DisposableBean

注意:Aware接口的执行时机在(初始化/销毁)注解和接口之间

Scope
Scope的类型有哪些
  • singleton:单例对象
  • prototype:原型,每次获取会创建一个新的对象
  • request:作用于Web的request作用域中,每次请求会创建一个新的对象
  • session:作用于Web的session作用域中,每个新的会话会创建一个新的对象
  • application:作用域Web的applicaion作用域中
Scope的销毁
  • singleton:当容器关闭时销毁
  • prototype:不受容器管控
  • request:每次请求结束
  • session:会话超时
  • application:没有被spring正确实现

指定Bean的作用域可以使用@Scope注解指定

Web作用域演示

在这里插入图片描述

Scope失效问题

当在singleton作用域调用其他作用域的对象时会导致其他作用域失效,原因是因为依赖注入只会注入一次

解决方案

  • 在注入的对象上使用@Lazy注解,SpringIoC容器会在启动的时候实例化所有单实例 bean 。如果我们想要实现 Spring 在启动的时候延迟加载 bean,即在首次调用bean的时候再去执行初始化
  • 指定@Scope(proxyMode=ScopedProxyMode.TARGET_CLASS)
  • 使用ObjectFactory来创建对象
  • 使用ApplicationContext获取对象

原理:都是延迟其他scope bean的获取时机

查看全文

99%的人还看了

猜你感兴趣

版权申明

本文"Spring底层原理(五)":http://eshow365.cn/6-31671-0.html 内容来自互联网,请自行判断内容的正确性。如有侵权请联系我们,立即删除!