Java中“常量”实为public static final字段,需同时满足public、static、final;final仅保证引用不可变,不保证对象内容不可变;多值常量优先用enum。

Java里定义常量必须用public static final
Java没有真正的“常量”关键字,所谓“常量”只是约定俗成的public static final字段。它同时满足三个条件:对外可见(public)、属于类而非实例(static)、不可重新赋值(final)。
常见错误是只写final或漏掉static:
-
final int MAX_RETRY = 3;→ 实例字段,每个对象都有一份,不是常量 -
public final int MAX_RETRY = 3;→ 仍是实例字段,无法通过类名直接访问 -
public static int MAX_RETRY = 3;→ 可被修改,不安全
正确写法:
public class Config {
public static final int MAX_RETRY = 3;
public static final String API_BASE_URL = "https://api.example.com";
}
final变量初始化必须在声明时或构造器中完成
final修饰的变量(包括实例字段、局部变量、参数)一旦声明,就必须在对象构造完成前(实例字段)或作用域结束前(局部变量)被明确赋值一次,且仅一次。
立即学习“Java免费学习笔记(深入)”;
典型报错:variable might not have been initialized 或 cannot assign a value to final variable。
- 实例
final字段:只能在声明处赋值,或在所有构造器中显式赋值(不能靠条件分支“可能”赋值) - 局部
final变量:必须在首次使用前赋值,且不能在循环或条件块中“可能未赋值” - 方法参数加
final:仅禁止在方法体内重新赋值该参数引用,不影响对象内部状态
反例(编译失败):
public class User {
private final String name;
public User(boolean useDefault) {
if (useDefault) {
this.name = "default";
}
// 缺少 else 分支 → 编译报错
}
}
字符串字面量和final引用的区别很关键
final String s = "hello"; 并不表示字符串内容不可变,而是s这个引用不能再指向别的对象。由于String本身是不可变类,所以效果上安全;但对可变类型(如ArrayList),final只锁住引用,不锁住内容。
-
final List→ 合法,但list = new ArrayList(); list.add("x")完全允许 - 要真正“不可变集合”,得用
Collections.unmodifiableList()或List.of()(Java 9+) -
final数组(如final int[] arr = {1,2};)同样只禁止重赋值arr,不限制arr[0] = 99
真正安全的常量集合示例:
public class Constants {
public static final List SUPPORTED_LOCALES =
List.of("en", "zh", "ja");
// 不可添加、删除、修改,且引用不可变
}
枚举类比public static final更适合多值常量场景
当常量是一组有语义关联的固定取值(如状态码、协议类型、权限级别),用enum比一堆static final字段更安全、更易维护。
- 枚举实例天然
final且单例,无法被外部构造 - 支持方法、字段、构造器,可封装行为(比如
HTTP_STATUS.OK.getCode()) - IDE能提示所有可选值,避免拼写错误;
switch支持完备性检查(Java 14+) - 序列化/反序列化更可靠,不会因字段名变更失效
对比:
// 不推荐:分散、无约束、易错
public static final int STATUS_PENDING = 0;
public static final int STATUS_PROCESSING = 1;
public static final int STATUS_DONE = 2;
// 推荐:集中、类型安全、可扩展
public enum OrderStatus {
PENDING(0), PROCESSING(1), DONE(2);
private final int code;
OrderStatus(int code) { this.code = code; }
public int getCode() { return code; }
}
容易忽略的是:final不传递不可变性,也不保证线程安全——如果一个final字段引用了可变对象,且该对象被多个线程共享修改,仍需额外同步。










