类加载与初始化机制

核心逻辑(适用于大多数OOP语言)​

执行顺序铁律:​
父类静态 → 子类静态 → 父类实例 → 子类实例

(静态数据只初始化一次,实例数据每次new都执行)


​**📊 通用执行顺序流程图**

css 复制代码
mermaid
graph TD
    A[父类静态成员初始化] --> B[子类静态成员初始化]
    B --> C[父类实例成员初始化]
    C --> D[父类构造函数]
    D --> E[子类实例成员初始化]
    E --> F[子类构造函数]

​**🛠️ 关键概念解析**

1. 静态 vs 非静态

特性 静态成员 非静态成员
初始化时机 类加载时(仅一次) 每次实例化时
内存分配 全局唯一 每个实例独立一份
访问方式 类名.成员 实例.成员

2. 类加载阶段(语言无关)​

  1. 加载 :读取类定义(如.class/.dll文件)
  2. 验证:检查语法/安全性
  3. 准备:为静态变量分配内存(赋默认值)
  4. 解析:处理符号引用
  5. 初始化:执行静态代码和静态变量赋值

3. 实例化阶段

  1. 分配实例内存
  2. 初始化实例变量
  3. 执行构造函数

​**🌰 伪代码示例**

场景:父类与子类初始化

scala 复制代码
plaintext
class Parent:
    static:
        print("父类静态初始化")
    
    instance:
        print("父类实例初始化")
    
    constructor():
        print("父类构造函数")

class Child extends Parent:
    static:
        print("子类静态初始化")
    
    instance:
        print("子类实例初始化")
    
    constructor():
        print("子类构造函数")

// 测试
obj = new Child()

输出结果

复制代码
markdown
父类静态初始化
子类静态初始化
父类实例初始化
父类构造函数
子类实例初始化
子类构造函数
  1. 默认值优先

    所有静态变量会先赋默认值(如int=0),再执行显式赋值:

    ini 复制代码
    java
    static int a = 5;
    static { a = 10; }
    // 最终a=10(默认0 → 赋5 → 赋10)

​**✅ 终极总结**

  1. 类加载时:静态成员(变量/代码块)按父→子顺序初始化

  2. 实例化时

    • 父类实例成员 → 父类构造
    • 子类实例成员 → 子类构造
  3. 静态只一次:无论创建多少实例,静态数据仅初始化一次

需要注意 类加载的时候会上锁 其他线程会阻塞

相关推荐
IT_陈寒11 分钟前
Redis内存爆了,原来我漏掉了这个致命配置
前端·人工智能·后端
fliter1 小时前
最后一块拼图:用 bitvec 构造 IPv4 包,真正做出自己的 Ping
后端
fliter2 小时前
用 Rust 解析并生成 ICMP 包:checksum、nom 与 cookie-factory
后端
蝎子莱莱爱打怪2 小时前
XZLL-IM干货系列 03|消息 ID 设计:一个 UUID 搞不定的事,我用两个 ID 解决了
后端·面试·开源
fliter2 小时前
从 panic 到 Result:用 Rust 重新整理一个 ping 项目的错误处理
后端
森蓝情丶3 小时前
我给 AI 搭了个法庭:一个前端仔的 LangGraph 实战全记录
前端·后端
JensCS猿3 小时前
从 Spring Boot 回看 SSM 框架:手动挡与自动挡的驾驶哲学
后端
爱勇宝3 小时前
干了近 8 年,一夜之间被裁:AI 时代,程序员最该害怕的不是 AI
前端·后端·程序员
科米米3 小时前
嵌入式日志模块
后端
血小溅3 小时前
三大 AI 编码框架深度对比:GSD vs OpenSpec vs Superpowers
人工智能·后端