static变量属类所有、共享内存,类加载时初始化且仅一次;static方法只能访问static成员,不依赖对象;static块用于一次性复杂初始化;static内部类不持外部类引用,防内存泄漏。

static变量属于类而非实例,所有对象共享同一份内存
Java中用static修饰的变量在类加载时就分配内存,且只分配一次。它不依赖对象创建,即使没有new任何实例,也能通过类名.变量名访问。
- 修改
static变量会影响所有对象——比如计数器、配置开关、缓存容器 - 不能在
static上下文中直接访问非static成员(会报non-static variable cannot be referenced from a static context) - 初始化顺序:静态变量按声明顺序初始化,早于构造方法执行
public class Counter {
public static int count = 0; // 所有实例共用
private int id;
public Counter() {
this.id = ++count; // 每次new都让count自增
}
}
static方法只能访问static成员,也不能用this/super
static方法属于类本身,不绑定具体对象,因此无法使用this或super,也不能调用非static字段和方法。
- 常见用途:工具方法(如
Math.max())、工厂方法(如LocalDateTime.now())、main入口 - 子类可以隐藏(hide)父类的
static方法,但不是重写(override)——多态不生效 - 不能被
abstract修饰(无意义),也不能是final以外的访问修饰符组合(如private static合法,protected abstract static非法)
public class StringUtils {
public static boolean isBlank(String s) {
return s == null || s.trim().isEmpty();
}
// ❌ 编译错误:Cannot use 'this' in a static context
// public static void printId() { System.out.println(this.id); }
}
static块用于类加载时的一次性初始化
static代码块在类第一次被加载(如首次new、首次调用static成员、首次反射获取Class)时执行,且仅执行一次。适合做复杂静态资源初始化。
- 多个
static块按出现顺序执行 - 不能抛出受检异常(checked exception),除非用
try-catch包裹 - 比静态变量初始化更灵活——可写逻辑、读配置、建连接池等
public class ConfigLoader {
public static final Map CONFIG;
static {
Properties props = new Properties();
try (InputStream is = ConfigLoader.class.getResourceAsStream("/app.properties")) {
props.load(is);
CONFIG = new HashMap<>(props::forEach);
} catch (IOException e) {
throw new ExceptionInInitializerError(e); // 静态块异常会导致Class.forName失败
}
}
}
static内部类不持有外部类引用,避免内存泄漏
普通内部类隐式持有外部类实例引用,而static内部类不持有没有引用,因此不会阻止外部类被GC回收。这是它最常被忽略却关键的用途。
立即学习“Java免费学习笔记(深入)”;
- 必须显式通过外部类名访问其
static成员,不能访问非static字段 - 适合定义工具类、枚举辅助类、Builder模式中的静态构建器
- 若误删
static导致内部类变成非静态,又在长生命周期对象(如单例、线程池任务)中持有它,可能造成外部类实例无法释放
public class CacheManager {
private final String cacheName = "default";
// ✅ 静态内部类,不引用CacheManager实例
public static class Builder {
private String name;
public Builder setName(String name) {
this.name = name;
return this;
}
public CacheManager build() {
CacheManager cm = new CacheManager();
// ... 可通过其他方式设置,但不依赖cm实例状态
return cm;
}
}
}
真正容易出问题的地方不在语法,而在生命周期认知:static变量长期驻留堆内存,static内部类看似轻量,一旦它被静态集合持有(比如static List POOL ),而Builder又意外捕获了外部引用(比如忘了加static),就会悄悄拖住整个外部类实例。










