静态成员属类、共享且类加载时初始化;实例成员属对象、独立且每次new时创建;静态方法不能访问实例成员;静态块先于实例块执行;静态内部类不持外部引用,非静态则持有;静态集合易致内存泄漏。

静态成员属于类,实例成员属于对象
静态成员(static 字段或方法)在类加载时就分配内存,且只有一份,被该类所有实例共享;实例成员每次 new 一个对象时才创建一份,彼此独立。这意味着修改某个对象的实例字段不会影响其他对象,但修改静态字段会直接影响所有对象(包括尚未创建的)。
- 静态成员可通过类名直接访问:
MyClass.count,无需实例 - 实例成员必须通过对象访问:
obj.name,否则编译报错non-static variable xxx cannot be referenced from a static context - 静态方法内部不能直接调用实例方法或访问实例字段——因为此时可能根本没有实例存在
静态块和实例初始化块的执行时机不同
static { ... } 块在类第一次被加载(如首次 new、首次访问静态成员、首次反射加载)时执行,且仅执行一次;而实例初始化块({ ... })每次构造对象时都执行,且在构造函数体之前运行。
- 静态块适合做类级资源初始化(如加载配置、注册驱动)
- 实例块适合做多个构造函数共用的初始化逻辑,避免重复代码
- 若类中既有静态块又有实例块,静态块一定先于任何实例块执行
public class Example {
static { System.out.println("静态块"); }
{ System.out.println("实例块"); }
public Example() { System.out.println("构造函数"); }
}
执行 new Example() 输出顺序为:静态块 → 实例块 → 构造函数
静态内部类与非静态内部类的访问权限差异
静态内部类(static class Inner)不持有外部类的引用,因此不能直接访问外部类的实例成员;而非静态内部类(默认)隐式持有一个外部类实例的引用,可自由访问其所有成员(包括私有实例字段)。
立即学习“Java免费学习笔记(深入)”;
- 静态内部类可直接通过
Outer.Inner实例化,无需外部类对象 - 非静态内部类必须依附于外部类实例:
outer.new Inner()或在外部类方法内用new Inner() - 误将内部类声明为
static后又试图访问this.field,编译器会报错cannot reference a non-static field from a static context
内存泄漏风险常来自静态成员持有对象引用
静态集合(如 static List)若长期持有 Activity、Context、View 等 Android 对象,或 Web 中的 HttpServletRequest,会导致这些对象无法被 GC 回收,引发内存泄漏。
- 避免用静态集合缓存生命周期短的对象
- 若必须缓存,优先使用弱引用:
static Map> - 注意日志框架中静态
Logger不会造成泄漏(它不持有业务对象),但静态Map很容易成为泄漏源头
最隐蔽的问题是:静态成员的生命周期与类加载器绑定,只要类没卸载,它就一直存在——这点在热部署、OSGi、Web 容器中尤其关键。










