静态成员属于类本身而非实例,JVM在类初始化阶段为其分配唯一内存,所有实例共享;调用不依赖对象,由声明类型决定隐藏行为,与对象无关。

静态成员不属于任何对象,而是属于类本身。
静态成员的归属主体是类,不是实例
在Java中,static修饰的字段、方法、代码块或内部类,都直接绑定到类(Class)这个运行时对象上,而不是绑定到某个具体的实例对象。JVM在类加载阶段(准确说是初始化阶段)就为静态成员分配内存,并且只分配一次。所有该类的实例共享同一份静态成员数据。
- 每个对象都有自己的非静态字段副本,但所有对象访问的静态字段都是同一个内存地址
- 调用静态方法不需要创建对象,例如 Math.abs(-5)、Integer.parseInt("123")
- 即使一个类从未被实例化,只要它被加载并初始化,其静态成员就已存在
静态成员与类加载过程强相关
静态成员的生命周期由类的加载、链接、初始化三阶段决定。特别是“初始化”阶段,JVM会执行类的静态初始化块和静态字段的显式赋值语句。这个过程只发生一次,且是线程安全的(由JVM保证)。
- 首次主动使用类时触发初始化:比如 new 实例、调用静态方法、访问静态字段(非final常量除外)、反射、子类初始化等
- 父类先于子类初始化;接口不触发父接口的初始化(除非直接引用)
- 静态字段若被声明为 final + 编译期常量(如 public static final int MAX = 100;),则不会触发类初始化,其值在编译时就被内联到调用处
静态成员不可被重写,但可被隐藏
子类可以定义与父类同名的静态成员(字段或方法),但这不是多态意义上的“重写”,而是“隐藏”。调用哪个静态成员,取决于**引用变量的声明类型**,而非实际对象类型。
立即学习“Java免费学习笔记(深入)”;
- 例如:Parent p = new Child(); p.staticMethod(); 调用的是 Parent 的静态方法
- 而 Child.staticMethod(); 则调用 Child 的版本
- 这种行为与实例方法的动态绑定截然不同,也说明静态成员本质上与对象无关
常见误区提醒
不要误以为“通过对象访问静态成员”就意味着它属于该对象。语法上允许 obj.staticField 或 obj.staticMethod(),但这只是编译器的语法糖,实际仍按类名去解析。
- 编译后,上述写法会被自动替换为 ClassName.staticField
- 如果对象引用为 null,静态调用依然成功(因为根本不依赖实例)
- IDE通常会警告“静态成员应通过类名访问”,正是为了强调其归属本质










