Spring IOC详解及Bean生命周期详细过程,看完直接吊打面试官

1、实例化前置


实例化前置使用的是 InstantiationAwareBeanPostProcessor.postProcessBeforeInstantiation(Class<?> beanClass, String beanName) 方法,方法里有2个参数,分别是beanClass和beanName,顾名思义,就是对在对象实例化之前对bean对象的class信息进行修改或者扩展,以达到我们想要的功能,它的底层是动态代理AOP技术实现的;且是bean生命周期中最先执行的方法;

返回非空:返回值是Object类型,这意味着我们可以返回任何类型的值,由于这个时候目标对象还未实例化,所以这个返回值可以用来代替原本该生成对象的目标对象的实例,也就是说,如果返回了非空的值,那么以后我们需要用到这个bean的时候,拿到的就现在返回的对象了,也就不会去走第二步去实例化对象了;

返回空(null)值:默认也是返回null值的,那么就直接返回,接下来会调用doCreateBean方法来实例化对象;

2、实例化对象


doCreateBean方法创建实例,用反射技术创建,这个没什么好说的,只是相当于new了一个对象出来而已,但需要注意的是,这个时候只是将对象实例化了,对象内的属性还未设置;

3、实例化后置


方法名称: InstantiationAwareBeanPostProcessor.postProcessAfterInstantiation(Object bean, String beanName)

在目标对象实例化之后调用,这个时候对象已经被实例化,但是该实例的属性还未被设置,都是null。因为他的返回值是决定要不要调用postProcessPropertyValues方法中的一个因素(因为还有一个因素是mbd.getDependencyCheck());

返回false :如果该方法返回false,并且不需要check,那么postProcessPropertyValues就会被忽略不执行;

返回true : 如果返回true,postProcessPropertyValues就会被执行

4、属性修改


方法名称 :InstantiationAwareBeanPostProcessor.PropertyValues postProcessPropertyValues(PropertyValues pvs, PropertyDescriptor[] pds, Object bean, String beanName)

此方法可对属性值进行修改,修改范围包括添加、修改、删除操作;,如果实例化后置 postProcessAfterInstantiation() 方法返回false,那么该方法不会被调用;

5、给用户属性赋值


用户属性指的是用spring 的人自定义的bean对象属性,像 User、Student、Teacher 、UserService、IndexService 这类的对象都是自定义bean对象,第5步主要给这类属性进行赋值操作,使用的是  AbstractAutowireCapableBeanFactory.populateBean() 方法进行赋值;

6、给容器属性赋值


容器属性其实就是容器自带的属性,这些属性都是spring本来就有的;可以肯定的是,它们都是 Aware 接口的实现类,主要有以下实现类,我已经将它们的执行顺序都排列好了,

image.png

我们先看看怎么用,然后再来讲解每个Aware的作用;上代码

package com.Spring.Boot.init.aware;

import org.springframework.beans.BeansException;

import org.springframework.beans.factory.BeanClassLoaderAware;

import org.springframework.beans.factory.BeanFactory;

import org.springframework.beans.factory.BeanFactoryAware;

import org.springframework.beans.factory.BeanNameAware;

import org.springframework.context.*;

import org.springframework.context.annotation.ImportAware;

import org.springframework.context.weaving.LoadTimeWeaverAware;

import org.springframework.core.env.Environment;

import org.springframework.core.io.ResourceLoader;

import org.springframework.core.type.AnnotationMetadata;

import org.springframework.instrument.classloading.LoadTimeWeaver;

import org.springframework.stereotype.Component;

import org.springframework.util.StringValueResolver;

import org.springframework.web.context.ServletContextAware;

import javax.servlet.ServletContext;

@Component

public class AllAwareInterface implements BeanNameAware, BeanClassLoaderAware,

BeanFactoryAware, EnvironmentAware, EmbeddedValueResolverAware,

ResourceLoaderAware, ApplicationEventPublisherAware, MessageSourceAware,

ApplicationContextAware, ServletContextAware, LoadTimeWeaverAware, ImportAware {

@Override

public void setBeanName(String name) {

// BeanNameAware作用:让Bean对Name有知觉

//这个方法只是简单的返回我们当前的beanName,听官方的意思是这个接口更多的使用在spring的框架代码中,实际开发环境应该不建议使用

System.out.println(“1 我是 BeanNameAware 的 setBeanName 方法 —参数:name,内容:”+ name);

}

@Override

public void setBeanClassLoader(ClassLoader classLoader) {

System.out.println(“2 我是 BeanClassLoaderAware 的 setBeanClassLoader 方法”);

}

@Override

public void setBeanFactory(BeanFactory beanFactory) throws BeansException {

// 注意: 如果使用 @Configuration 注解的话,setBeanFactory方法会执行2次,

System.out.println(“3 我是 BeanFactoryAware 的 setBeanFactory 方法”);

}

@Override

public void setEnvironment(Environment environment) {

System.out.println(“4 我是 EnvironmentAware 的 setEnvironment 方法”);

}

@Override

public void setEmbeddedValueResolver(StringValueResolver stringValueResolver) {

System.out.println(“5 我是 EmbeddedValueResolverAware 的 setEmbeddedValueResolver 方法”);

}

@Override

public void setResourceLoader(ResourceLoader resourceLoader) {

System.out.println(“6 我是 ResourceLoaderAware 的 setResourceLoader 方法”);

}

@Override

public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {

System.out.println(“7 我是 ApplicationEventPublisherAware 的 setApplicationEventPublisher 方法”);

}

@Override

public void setMessageSource(MessageSource messageSource) {

System.out.println(“8 我是 MessageSourceAware 的 setMessageSource 方法”);

}

@Override

public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {

System.out.println(“9 我是 ApplicationContextAware 的 setApplicationContext 方法”);

}

@Override

public void setServletContext(ServletContext servletContext) {

System.out.println(“10 我是 ServletContextAware 的 setServletContext 方法”);

}

@Override

public void setLoadTimeWeaver(LoadTimeWeaver loadTimeWeaver) {

//LoadTimeWeaver 简称LTW,LTW是AOP的一种实现方式,此方法是为了获取Aop织入的对象,使用的织入方式是:类加载期织入,

// 一般的aop都是运行期织入,就是在运行的时候才进行织入切面方法,但是LTW是在类加载前就被织入了,也就是class文件在jvm加载之前进行织入切面方法

// 只有在使用 @EnableLoadTimeWeaving 或者存在 LoadTimeWeaver 实现的 Bean 时才会调用,顺序也很靠后

System.out.println(“11 我是 LoadTimeWeaverAware 的 setLoadTimeWeaver 方法”);

}

@Override

public void setImportMetadata(AnnotationMetadata annotationMetadata) {

//只有被其他配置类 @Import(XX.class) 时才会调用,这个调用对 XX.class 中的所有 @Bean 来说顺序是第 1 的。

System.out.println(“12 我是 ImportAware 的 setImportMetadata 方法”);

}

}

复制代码

启动spring后的控制台打印的部分结果如下:

image.png

可以看到它们的输出结果按照顺序依次排列打印出来了,这就是它的标准顺序了;接下来我们了解下它们的具体作用

6.1  BeanNameAware.setBeanName()

这个方法只是简单的返回我们当前的beanName,听官方的意思是这个接口更多的使用在spring的框架代码中,实际开发环境应该不建议使用

6.2  BeanClassLoaderAware.setBeanClassLoader()

获取Bean的类装载器,

6.3 BeanFactoryAware.setBeanFactory()

获取bean工厂,beanFactory让你可以不依赖注入方式,随意的读取IOC容器里面的对象,不过beanFactory本身还是要注入的。

需要注意的是,一般情况下我们都用  @Component 注解,如果使用 @Configuration 注解的话,setBeanFactory方法会执行2次;

6.4 EnvironmentAware.setEnvironment()

实现了EnvironmentAware接口重写setEnvironment方法后,在工程启动时可以获得application.properties 、xml、yml 的配置文件配置的属性值。

6.5  EmbeddedValueResolverAware.setEmbeddedValueResolver()

通常我们使用@Value注解来获取properties 和 yml 文件中的值,每个类中都要使用@Value也很繁琐,实现EmbeddedValueResolverAware接口后就方便多了。用法也跟@Value一样,需要用${}包裹住;

@Component

public class PropertiesUtil implements EmbeddedValueResolverAware {

@Override

public void setEmbeddedValueResolver(StringValueResolver stringValueResolver) {

System.out.println(stringValueResolver.resolveStringValue(“${logging.file}”));

}

}

复制代码

6.6 ResourceLoaderAware.setResourceLoader()

Spring ResourceLoader为我们提供了一个统一的getResource()方法来通过资源路径检索外部资源。从而将资源或文件(例如文本文件、XML文件、属性文件或图像文件)加载到Spring应用程序上下文中的不同实现 ,其实说白了,就是用来加载外部资源的;方法中有个参数:ResourceLoader ,这个参数其实就是ApplicationContext(spring 的上下文对象);可直接强转;

package org.crazyit.app.service;

import org.springframework.context.ResourceLoaderAware;

import org.springframework.core.io.ResourceLoader;

public class TestBean implements ResourceLoaderAware{

public void setResourceLoader(ResourceLoader resourceLoader) {

// 可直接强转为 ApplicationContext

ApplicationContext context = (ApplicationContext) resourceLoader;

System.out.println(“6 我是 ResourceLoaderAware 的 setResourceLoader 方法”);

}

}

复制代码

并且我们可以指定不同的前缀来创建路径以从不同位置加载资源

image.png

6.7 ApplicationEventPublisherAware.setApplicationEventPublisher();

ApplicationEventPublisherAware是一个事件发布器的接口,使用这个接口,我们自己的 Service 就拥有了发布事件的能力。用户注册后,不再是显示调用其他的业务 Service,而是发布一个用户注册事件。那么在这里是发布事件,那就肯定有监听事件的接口,这个接口叫做 ApplicationListener ,只要实现 ApplicationListener 接口就可以接受发布的事件了,接下来我们写一个示例来模拟发布事件和监听事件;

先创建一个实体类,用来存储发布的事件内容   StringEvent.java

package com.Spring.Boot.init.listener.eventModel;

import org.springframework.context.ApplicationEvent;

//事件监听对象

public class StringEvent extends ApplicationEvent {

private String str;

// 构造函数

public StringEvent(Object source) {

super(source);

str = source.toString();

}

// 获取字符串

public String getStr(){

return str;

}

}

复制代码

创建一个发布事件的类: ExtApplicationEventPublisherAware.java  ,实现 ApplicationEventPublisherAware  接口增加发布事件的功能;

package com.Spring.Boot.init.aware;

import com.Spring.Boot.init.listener.eventModel.StringEvent;

import org.springframework.context.ApplicationEventPublisher;

import org.springframework.context.ApplicationEventPublisherAware;

import org.springframework.stereotype.Component;

/**

  • 发布事件

*/

@Component

public class ExtApplicationEventPublisherAware implements ApplicationEventPublisherAware {

@Override

public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) {

System.out.println(“发布事件,事件对象为 StringEvent ,内容为 :1234”);

StringEvent stringEvent = new StringEvent(“1234”);

// 发布事件 ,发布后会在 ApplicationListener.onApplicationEvent()方法进行捕获;

applicationEventPublisher.publishEvent(stringEvent); // 发布事件

}

}

复制代码

在创建一个事件监听器:  EventListener.java ,用来监听所有发布的事件;

package com.Spring.Boot.init.listener;

import com.Spring.Boot.init.listener.eventModel.StringEvent;

import org.springframework.context.ApplicationEvent;

import org.springframework.context.ApplicationListener;

import org.springframework.stereotype.Component;

//事件监听器

@Component

public class EventListener implements ApplicationListener {

@Override

public void onApplicationEvent(StringEvent o) {

System.out.println(“监听到事件,内容:”+o.getStr());

}

}

复制代码

接下来,运行spring项目,看看打印的结果如下,到这里,事件的发布和监听就完成了;

image.png

6.8 MessageSourceAware.setMessageSource()

国际化消息通知操作

6.9 ApplicationContextAware.setApplicationContext()

ApplicationContextAware 主要用来全局获取 ApplicationContext 上下文,ApplicationContext其实就是容器,为此我们可以实现 ApplicationContextAware 接口来获取ApplicationContext容器对象;我们可以把它做成一个公共的静态类,这样可以在任意地方想拿就拿了,

package com.Spring.Boot.init.aware;

import org.springframework.beans.BeansException;

import org.springframework.context.ApplicationContext;

import org.springframework.context.ApplicationContextAware;

import org.springframework.context.annotation.Configuration;

import org.springframework.stereotype.Component;

@Component

public class ExtApplicationContextAware implements ApplicationContextAware {

/**

  • Spring容器会在加载完Spring容器后调用ApplicationContextAware.setApplicationContext方法

  • ApplicationContextAware 主要用来全局获取 ApplicationContext 上下文,

*/

private static ApplicationContext applicationContext;

@Override

public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {

if (ExtApplicationContextAware.applicationContext == null) {

ExtApplicationContextAware.applicationContext = applicationContext;

}

System.out.println(“ApplicationContext配置成功”);

System.out.println(“在普通类可以通过调用SpringBootBeanUtil.getApplicationContext()获取applicationContext对象”);

System.out.println(“applicationContext=“+ ExtApplicationContextAware.applicationContext +””);

}

/**

  • 获取applicationContext

  • @return

*/

public static ApplicationContext getApplicationContext() {

return applicationContext;

}

/**

  • 通过name获取 Bean.

  • @param name

  • @return

*/

public static Object getBean(String name) {

return getApplicationContext().getBean(name);

}

/**

  • 通过class获取Bean.

  • @param clazz

  • @return

*/

public static T getBean(Class clazz) {

return getApplicationContext().getBean(clazz);

}

/**

  • 通过name,以及Clazz返回指定的Bean

  • @param name

  • @param clazz

  • @return

*/

public static T getBean(String name, Class clazz) {

return getApplicationContext().getBean(name, clazz);

}

}

复制代码

当然,也可以直接注入,就像这样:

@Autowired

private ApplicationContext applicationContext;

复制代码

6.10 ServletContextAware.setServletContext()

通过实现ServletContextAware接口可获取servletContext,也就是servlet的上下文;

什么是ServletContext : WEB容器在启动时,它会为每个WEB应用程序都创建一个对应的ServletContext对象,它代表当前web应用。ServletConfig对象中维护了ServletContext对象的引用,开发人员在编写servlet时,可以通过ServletConfig.getServletContext方法获得ServletContext对象。

由于一个WEB应用中的所有Servlet共享同一个ServletContext对象,因此Servlet对象之间可以通过ServletContext对象来实现通讯。ServletContext对象通常也被称之为context域对象。

6.11 LoadTimeWeaverAware.setLoadTimeWeaver()

其实在调试的时候还有2个没打印出来,第11个就是 LoadTimeWeaver, 简称LTW,LTW是AOP的一种实现方式,此方法是为了获取Aop织入的对象,使用的织入方式是:类加载期织入,一般的aop都是运行期织入,就是在运行的时候才进行织入切面方法,但是LTW是在类加载前就被织入了,也就是class文件在jvm加载之前进行织入切面方法只有在使用 @EnableLoadTimeWeaving 或者存在 LoadTimeWeaver 实现的 Bean 时才会调用,顺序也很靠后;

6.12  ImportAware.setImportMetadata()

还有一个没打印的就是ImportAware接口,这个接口的方法只有被其他配置类 @Import(XX.class) 时才会调用,这个调用对 XX.class 中的所有 @Bean 来说顺序是第 1 的。

7、初始化前置


方法名称: BeanPostProcessor.postProcessBeforeInitialization()

在每一个 Bean 初始化之前执行的方法(有多少 Bean 调用多少次)

注意 : 启用该方法后,标注了@PostConstruct注解的方法会失效

8、初始化后置


方法名称: BeanPostProcessor.postProcessAfterInitialization()

在每一个 Bean 初始化之后执行的方法(有多少 Bean 调用多少次)

初始化前置和初始化后置的实现代码如下

package com.Spring.Boot.init;

import org.springframework.beans.BeansException;

import org.springframework.beans.factory.config.BeanPostProcessor;

import org.springframework.stereotype.Component;

@Component

public class ExtBeanPostProcessor implements BeanPostProcessor {

@Override

public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {

// 在每一个 Bean 初始化之前执行的方法(有多少 Bean 调用多少次)

// 注意 : 启用该方法后,标注了@PostConstruct注解的方法会失效

System.out.println(“初始化前置方法”);

return null;

}

@Override

public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {

在每一个 Bean 初始化之后执行的方法(有多少 Bean 调用多少次)

System.out.println(“初始化后置方法”);

return null;

}

}

复制代码

9、执行初始化方法


初始化方法有三个,分别是 添加了@PostConstruct 注解的方法、实现InitializingBean接口、在@bean注解上添加 initMethod属性;我们一个个讲

10、初始化方法一:@PostConstruct


在bean对象内添加@PostConstruct 注解后即可实现初始化的功能,被@PostConstruct修饰的方法会在构造函数之后,init()方法之前运行。 有多个则会执行多次;

注意: 如果spring 实现了 BeanPostProcessor接口的postProcessBeforeInitialization() 方法,也就是12的初始后置方法,那么@PostConstruct注解会失效;

代码示例

package com.Spring.Boot.init;

import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;

// @PostConstruct注解

@Component

public class ExtPostConstruct {

/**

  • 被@PostConstruct修饰的方法会在构造函数之后,init()方法之前运行。如果有多个则会执行多次

  • 注意: 如果spring 实现了 BeanPostProcessor接口的postProcessBeforeInitialization方法,该@PostConstruct注解会失效

*/

@PostConstruct

public void init() {

System.out.println(“第一个init…”);

}

// 有多个会执行多次

@PostConstruct

public void init1() {

System.out.println(“第二个init1…”);

}

}

复制代码

11、InitializingBean.afterPropertiesSet()


spring 初始化方法之一,作用是在BeanFactory完成属性设置之后,执行自定义的初始化行为。

执行顺序:在initMethod之前执行,在@PostConstruct之后执行

代码示例

package com.Spring.Boot.init;

import org.springframework.beans.factory.InitializingBean;

import org.springframework.context.annotation.Configuration;

import org.springframework.stereotype.Component;

@Component

public class ExtInitializingBean implements InitializingBean {

@Override

public void afterPropertiesSet() throws Exception {

// 一个 InitializingBean 执行一次

// spring 初始化方法,作用是在BeanFactory完成属性设置之后,执行自定义的 初始化行为.

// 执行顺序:在initMethod之前执行,在@PostConstruct之后执行

System.out.println(“InitializingBean”);

}

}

复制代码

12、init-method


bean 配置文件属性 init-method 用于在bean初始化时指定执行方法,用来替代继承 InitializingBean接口,

注意的一点是只有一个类完整的实例被创建出来后,才能走初始化方法。

示例代码,先定义一个类: BeanTest.java ,在类中定义一个初始化方法 initMethod_1()

package com.Spring.Boot.init.bean;

public class BeanTest {

// 将要执行的初始化方法

public void initMethod_1(){

System.out.println(“我是beanTest的init方法”);

}

}

复制代码

xml 配置方式

<bean id="beanTest" class="com.BeanTest" init-method="init"></bean> 注解配置方式

package com.Spring.Boot.init;

import com.Spring.Boot.init.bean.BeanTest;

import org.springframework.context.annotation.Bean;

import org.springframework.stereotype.Component;

@Component()

public class InitMethod {

// 在@Bean注解上添加initMethod属性,指向类中的 initMethod_1 执行初始化方法

@Bean(initMethod = “initMethod_1”)

public BeanTest getBeanTest(){
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数Java工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

看完美团、字节、腾讯这三家的面试问题,是不是感觉问的特别多,可能咱们又得开启面试造火箭、工作拧螺丝的模式去准备下一次的面试了。

开篇有提及我可是足足背下了1000道题目,多少还是有点用的呢,我看了下,上面这些问题大部分都能从我背的题里找到的,所以今天给大家分享一下互联网工程师必备的面试1000题

注意不论是我说的互联网面试1000题,还是后面提及的算法与数据结构、设计模式以及更多的Java学习笔记等,皆可分享给各位朋友

最新“美团+字节+腾讯”一二三面问题,挑战一下你能走到哪一面?

互联网工程师必备的面试1000题

而且从上面三家来看,算法与数据结构是必备不可少的呀,因此我建议大家可以去刷刷这本左程云大佬著作的《程序员代码面试指南 IT名企算法与数据结构题目最优解》,里面近200道真实出现过的经典代码面试题

最新“美团+字节+腾讯”一二三面问题,挑战一下你能走到哪一面?

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

因此收集整理了一份《2024年Java开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。[外链图片转存中…(img-fx3gmlUO-1713395536098)]

[外链图片转存中…(img-dwQJrg4C-1713395536098)]

[外链图片转存中…(img-W9uSCyP4-1713395536099)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Java开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注Java获取)

img

最后

看完美团、字节、腾讯这三家的面试问题,是不是感觉问的特别多,可能咱们又得开启面试造火箭、工作拧螺丝的模式去准备下一次的面试了。

开篇有提及我可是足足背下了1000道题目,多少还是有点用的呢,我看了下,上面这些问题大部分都能从我背的题里找到的,所以今天给大家分享一下互联网工程师必备的面试1000题

注意不论是我说的互联网面试1000题,还是后面提及的算法与数据结构、设计模式以及更多的Java学习笔记等,皆可分享给各位朋友

[外链图片转存中…(img-pfBz11vB-1713395536099)]

互联网工程师必备的面试1000题

而且从上面三家来看,算法与数据结构是必备不可少的呀,因此我建议大家可以去刷刷这本左程云大佬著作的《程序员代码面试指南 IT名企算法与数据结构题目最优解》,里面近200道真实出现过的经典代码面试题

[外链图片转存中…(img-WQq4MgDA-1713395536099)]

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

  • 26
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值