封装是通过访问修饰符控制访问权限,隐藏内部实现并暴露安全接口。核心在于合理使用private、getter/setter、不可变返回值、防御性拷贝及接口隔离,而非简单用class包裹代码。
Java 中的封装不是单纯用 class 把字段和方法塞进一个类里就完事。它的核心是:**通过访问修饰符(private、protected、public、默认包级)明确谁可以读、谁可以改、谁只能调用,从而把内部实现细节藏起来,只暴露安全可控的接口**。
常见错误是把所有字段都设成 public,或者只加了 private 却没配合理的 getter/setter —— 这等于上了锁却把钥匙焊在门把手上。
private 字段 + 无条件 public setter = 封装形同虚设return this.list)导致外部能直接修改内部状态 = 破坏封装边界不是每个 private 字段都必须配一对 getX() 和 setX()。是否提供、是否校验、是否返回副本,取决于这个字段在模型中的角色。
public class BankAccount {
private BigDecimal balance;
// ✅ 余额只允许通过 deposit/withdraw 修改,不暴露 setBalance
public void deposit(BigDecimal amount) {
if (amount.compareTo(BigDecimal.ZERO) > 0) {
this.balance = this.balance.add(amount);
}
}
// ✅ getBalance 返回不可变值(BigDecimal 本身不可变),安全
public BigDecimal getBalance() {
return this.balance; // 不用 new BigDecimal(balance)
}
// ❌ 不要这样:
// public void setBalance(BigDecimal balance) { this.balance = balance; }
// 外部可随意篡改,且无审计、无事件通知、无一致性检查
}
UnsupportedOperationException
Collections.unmodifiableList(this.items)
null 或占位符,避免日志/调试意外泄露protected 不代表“子类专用”,它允许:同一包内任意类访问 + 所有子类(即使跨包)访问。很多团队用 protected 暴露字段给测试类,结果导致业务模块也能直接依赖、修改这些“内部”状态。
包级(默认)访问看似安全,但只要类在同一 package 下,哪怕不属于同一模块,也能访问 —— 在多模块 Maven 项目中,这极易因包名巧合或重构疏忽打破封装边界。
@VisibleForTesting 注解 + package-private + 文档说明,而非降级为 protected
protected final 方法定义钩子点,而不是暴露 protected 字段protected 字段共享数据你以为封装好了,结果一传参、一链式调用,内部状态就漏了。典型场景:
ArrayList)并直接赋值给 private 字段 → 外部后续修改原列表,内部也跟着变list.add(...) 直接污染对象状态build() 返回的对象仍持有对 builder 内部可变状态的引用关键原则:**防御性拷贝(defensive copy)不是过度设计,而是封装的必要成本**。
public class Person {
private final List phones;
// ✅ 构造时复制
public Person(List
phones) {
this.phones = new ArrayList<>(Objects.requireNonNull(phones));
}
// ✅ getter 返回不可修改副本
public List getPhones() {
return Collections.unmodifiableList(phones);
}
}
封装真正的难点不在语法,而在每次添加一个 public 方法或暴露一个字段时,你是否清楚它会把哪块内部逻辑、哪些约束、哪些生命周期责任一起交出去。