Spring Bean初始化及@PostConstruc执行顺序示例详解

 更新时间:2025年07月24日 11:10:57   作者:基础不牢,地动山摇  
本文给大家介绍Spring Bean初始化及@PostConstruc执行顺序,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

1. Bean初始化执行顺序

  1. 构造函数 - 首先调用Bean的构造函数

  2. 依赖注入 - 完成所有依赖项的注入(@Autowired等)

  3. @PostConstruct方法 - 执行带有@PostConstruct注解的方法

  4. InitializingBean的afterPropertiesSet() - 如果Bean实现了InitializingBean接口

  5. 自定义init方法 - 执行通过@Bean(initMethod="...")或XML配置中init-method指定的方法

@Component
public class ExampleBean implements InitializingBean {
    public ExampleBean() {
        System.out.println("1. 构造函数");
    }
    @Autowired
    public void setDependency(SomeDependency dep) {
        System.out.println("2. 依赖注入");
    }
    @PostConstruct
    public void postConstruct() {
        System.out.println("3. @PostConstruct方法");
    }
    @Override
    public void afterPropertiesSet() {
        System.out.println("4. InitializingBean的afterPropertiesSet()");
    }
    @Bean(initMethod = "customInit")
    public void customInit() {
        System.out.println("5. 自定义init方法");
    }
}

2. 成员变量初始化顺序

在Java中,类的成员变量(非静态成员变量)的初始化时机取决于其定义方式(直接赋值、构造器赋值、初始化块等)以及是否涉及依赖注入(如Spring框架)。

2.1 普通Java类(非Spring环境)

成员变量的初始化顺序和时机如下:

(1) 默认初始化(即初始分配内存)

  • 时机:对象实例化时(new关键字调用构造器之前)。

  • 规则:所有成员变量先被赋予默认值(int0booleanfalse,引用类型→null等)。

(2) 显式初始化

  • 时机:紧随默认初始化之后,按代码中的声明顺序执行。

  • 方式

    • 直接赋值:

public class MyClass {
    private int a = 10;          // 显式赋值
    private String s = "hello";  // 显式赋值
}
  • 初始化块(代码块):
public class MyClass {
    private int x;
    {
        x = 20; // 初始化块赋值
    }
}

(3) 构造器初始化

  • 时机:在显式初始化之后,构造器最后执行。

  • 特点:构造器中的赋值会覆盖之前的默认值或显式赋值。

public class MyClass {
    private int value;
    public MyClass() {
        this.value = 30; // 构造器赋值
    }
}

(4)完整顺序

默认值 → 显式赋值/初始化块 → 构造器赋值

2.2 Spring管理的Bean(依赖注入场景)

在Spring中,成员变量的初始化分为依赖注入普通成员变量初始化两部分:

(1) 普通成员变量

  • 初始化规则与普通Java类一致(默认值 → 显式赋值 → 构造器)。

  • 示例

@Component
public class MyBean {
    private int count = 100;  // 显式赋值(Spring无关)
    public MyBean() {
        this.count = 200;     // 构造器覆盖
    }
}

(2) 依赖注入的成员变量

  • 时机:在Bean实例化后,由Spring容器通过反射或Setter方法注入。

    • 字段注入(@Autowired):
      在构造器和显式赋值之后,通过反射直接注入。

      @Component
      public class MyBean {
          @Autowired
          private Dependency dependency; // Spring在对象构造后注入
      }
    • 构造器注入
      在实例化时通过构造器参数注入(等同于普通Java的构造器赋值)。

      @Component
      public class MyBean {
          private final Dependency dependency;
          public MyBean(Dependency dependency) {
              this.dependency = dependency; // 构造时注入
          }
      }

(3) 生命周期回调的影响

  • @PostConstruct方法会在依赖注入完成后执行,此时所有成员变量(包括注入的依赖)已就绪:

    @Component
    public class MyBean {
        @Autowired
        private Dependency dependency;
        @PostConstruct
        public void init() {
            System.out.println(dependency); // 依赖已注入
        }
    }

(4)关键区别总结

场景成员变量初始化时机
普通Java类默认值 → 显式赋值/初始化块 → 构造器赋值
Spring Bean(字段注入)默认值 → 显式赋值 → 构造器 → 依赖注入 → @PostConstruct
Spring Bean(构造器注入)与普通Java类相同,依赖通过构造器参数传入

2.3 final成员变量

  • 必须声明时、初始化块或构造器中赋值,否则编译错误。

public class MyClass {
    private final int x = 10;  // 声明时赋值
    private final int y;
    { y = 20; }  // 初始化块赋值
    private final int z;
    public MyClass() {
        z = 30;  // 构造器赋值
    }
}

2.4 常见面试问题

Q1: 为什么构造器里的赋值能覆盖显式赋值?

因为构造器是最后执行的,可以修改之前的值。

Q2: 以下代码的输出是什么?

public class Test {
    private int x = 10;
    { x = 20; }
    public Test() {
        System.out.println(x);
        x = 30;
    }
    public static void main(String[] args) {
        new Test(); // 输出?
    }
}
--
答案:20(显式赋值和初始化块先执行,构造器最后执行)。

到此这篇关于Spring Bean初始化及@PostConstruc执行顺序的文章就介绍到这了,更多相关Spring Bean初始化内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • mybatis-plus查询方式,部分字段没有值

    mybatis-plus查询方式,部分字段没有值

    这篇文章主要介绍了mybatis-plus查询方式,部分字段没有值问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • java中的char占几个字节实例分析

    java中的char占几个字节实例分析

    这篇文章主要介绍了java中的char占几个字节实例分析的相关资料,需要的朋友可以参考下
    2017-04-04
  • spring-Kafka中的@KafkaListener深入源码解读

    spring-Kafka中的@KafkaListener深入源码解读

    本文主要通过深入了解源码,梳理从spring启动到真正监听kafka消息的这套流程,从spring启动开始处理@KafkaListener,本文结合实例流程图给大家讲解的非常详细,需要的朋友参考下
    2023-02-02
  • 使用Spring Retry实现业务异常重试

    使用Spring Retry实现业务异常重试

    在系统中经常遇到业务重试的逻辑,比如三方接口调用,timeout重试三遍,异常处理重试的兜底逻辑等,本文给大家介绍一下如何使用Spring Retry优雅的实现业务异常重试,需要的朋友可以参考下
    2024-01-01
  • 浅谈Java为什么只能单继承

    浅谈Java为什么只能单继承

    本文主要介绍了Java为什么只能单继承,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-03-03
  • 详解SpringBoot中实现依赖注入功能

    详解SpringBoot中实现依赖注入功能

    这篇文章主要介绍了详解SpringBoot中实现依赖注入功能,SpringBoot的实现方式基本都是通过注解实现的。有兴趣的可以了解一下。
    2017-04-04
  • 怎样通过分析GC日志来定位Java进程的内存问题

    怎样通过分析GC日志来定位Java进程的内存问题

    这篇文章主要介绍了怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2025-07-07
  • Java的RocketMq水平扩展及负载均衡详解

    Java的RocketMq水平扩展及负载均衡详解

    这篇文章主要介绍了Java的RocketMq水平扩展及负载均衡详解,RocketMQ是一个分布式具有高度可扩展性的消息中间件,本文旨在探索在broker端,生产端,以及消费端是如何做到横向扩展以及负载均衡的,需要的朋友可以参考下
    2024-01-01
  • Java8 中使用Stream 让List 转 Map使用问题小结

    Java8 中使用Stream 让List 转 Map使用问题小结

    这篇文章主要介绍了Java8 中使用Stream 让List 转 Map使用总结,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-06-06
  • MAVEN3.9.x中301问题及解决方法

    MAVEN3.9.x中301问题及解决方法

    本文主要介绍了使用MAVEN3.9.x中301问题及解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2025-01-01

最新评论