static修饰变量属类级别共享,所有实例共用同一内存;static方法无需实例、仅访问static成员;static块在类加载时执行一次;static内部类不依赖外部实例且无隐式引用。

static修饰变量:类级别共享,不依赖实例
被 static 修饰的变量属于类本身,而不是某个对象实例。所有实例共享同一份内存空间,修改任一实例的该变量值,其他实例看到的就是修改后的值。
常见错误现象:NullPointerException 不是因为 static 变量为空,而是误以为它会随实例初始化自动赋值;或者在多线程环境下未加同步,导致竞态修改。
- 不能在
static上下文中直接访问非static成员(如普通字段或方法),编译报错:non-static variable xxx cannot be referenced from a static context - 初始化时机在类加载时,早于任何构造器执行;可用静态代码块
static { ... }做复杂初始化 - 推荐用
private static final定义常量,避免意外修改
static修饰方法:无需实例即可调用
static 方法只能访问 static 字段和调用其他 static 方法,因为它不持有 this 引用——没有当前对象上下文。
典型使用场景:工具类(如 java.util.Collections)、工厂方法、主入口 public static void main(String[] args)。
立即学习“Java免费学习笔记(深入)”;
- 不能重写(override)
static方法,子类同名方法只是隐藏(hiding),调用取决于引用类型而非实际类型 - 不能用
this或super关键字 - 泛型方法若声明为
static,其类型参数必须独立于类泛型,例如public static是合法的List asList(T... a)
static代码块:控制类初始化逻辑
static 代码块在类第一次被加载时执行,且仅执行一次。适合做一次性资源准备,比如读配置、注册驱动、初始化缓存。
注意:多个 static 块按源码顺序执行;若抛出异常,类加载失败,后续所有对该类的引用都会触发 NoClassDefFoundError。
- 不能访问非
static成员 - 不能包含
return语句(除非在方法内) - 与
static字段初始化顺序一致:字段声明在前则先初始化,块在前则先执行
static内部类:脱离外部实例的嵌套类
static 内部类本质上是“寄生”在外部类命名空间下的独立类,不持有对外部类实例的隐式引用。因此它可以不依赖外部类实例而存在,也能定义自己的 static 成员。
容易踩的坑:误以为 static 内部类能直接访问外部类的非 static 字段——它不能,只能访问外部类的 static 成员。
- 创建实例不需要外部类对象:
new Outer.StaticInner() - 相比普通内部类,内存开销更低,无隐式引用泄漏风险
- 常用于 Builder 模式、工具辅助类(如
HashMap.Node)
static 看似简单,但真正难的是理解它和类加载、对象生命周期、内存可见性之间的耦合。尤其是多线程中对 static 变量的读写,不加同步或不满足 happens-before,结果可能完全不符合直觉。










