
本文介绍如何为java矩形类(rect)添加健壮的属性控制逻辑,确保width和length始终为正整数——对负值自动取绝对值,并通过构造器和setter方法统一约束入口,避免外部直接赋值导致的数据不一致。
在面向对象编程中,良好的封装不仅意味着将字段设为private,更在于主动控制数据的合法性与一致性。当前Rect类的问题在于:所有字段均为public(实际是包级访问,默认可被任意修改),且缺乏初始化校验逻辑,导致width或length可能被非法赋值(如0、负数),进而使面积、周长等计算结果失去业务意义。
✅ 正确做法:封装 + 校验 + 自动修正
首先,将所有字段声明为private,并提供受控的访问方式——推荐使用带校验逻辑的构造器和setter方法,而非依赖Math.abs()零散出现在各处(如原答案所提的this.width = Math.abs(width);虽能“修复”负值,但未解决根本问题:它无法阻止0或非法值传入,也未覆盖所有赋值路径)。
以下是重构后的完整代码:
public class Rect {
private int x;
private int y;
private int width;
private int length;
// 构造器:强制校验并初始化
public Rect(int x, int y, int width, int length) {
this.x = x;
this.y = y;
this.width = validatePositive(width);
this.length = validatePositive(length);
}
// 辅助方法:确保值为正整数,负数取绝对值,0则设为1(最小合法值)
private int validatePositive(int value) {
return Math.max(1, Math.abs(value));
}
// 提供安全的 setter 方法
public void setWidth(int width) {
this.width = validatePositive(width);
}
public void setLength(int length) {
this.length = validatePositive(length);
}
// getter 方法(可选,按需开放)
public int getWidth() { return width; }
public int getLength() { return length; }
public int getPerimeter() {
return 2 * (width + length);
}
public int getArea() {
return width * length;
}
public void move(int x, int y) {
this.x = x;
this.y = y;
}
public void changeSize(int n) {
this.width = validatePositive(n);
this.length = validatePositive(n);
}
public void print() {
int area = getArea();
int perimeter = getPerimeter();
System.out.printf("X: %d%n", this.x);
System.out.printf("Y: %d%n", this.y);
System.out.printf("Length: %d%n", this.length); // 注意:原代码中"Length"对应的是width字段,语义有歧义,此处按标准矩形命名修正
System.out.printf("Width: %d%n", this.width);
System.out.printf("Area: %d%n", area);
System.out.printf("Perimeter: %d%n", perimeter);
}
public static void main(String[] args) {
// 安全初始化:负值、零值均被自动修正
Rect r1 = new Rect(3, 4, -5, 0); // → width=5, length=1
r1.print();
// 运行时动态调整
r1.setWidth(-10);
r1.setLength(0);
System.out.println("\nAfter adjustment:");
r1.print();
}
}? 关键改进说明
- 字段私有化:private修饰符防止外部绕过逻辑直接修改,是封装的第一步;
- 统一校验入口:validatePositive()方法集中处理逻辑——Math.abs()仅解决符号问题,而Math.max(1, ...)确保最小值为1,真正满足“大于零”的业务要求;
- 构造器强制约束:对象创建即合规,杜绝“半初始化”状态;
- setter 显式可控:替代直接字段赋值,保证后续修改同样受控;
- 语义清晰化:原代码中length字段实际存储宽度,width存储长度,易引发混淆;本实现保留字段名但注释提醒,生产环境建议重命名为height/width或length/breadth以提升可读性;
- print()方法优化:不再依赖外部传入area/perimeter,改为内部调用,消除调用方耦合与潜在不一致风险。
⚠️ 注意事项
- 避免在main中直接写 r1.length = -3; —— 因字段已私有,编译会报错,这正是我们期望的保护机制;
- 若需支持浮点精度(如小数边长),应改用double类型,并采用Math.max(0.001, Math.abs(value))等策略,但需同步更新所有计算逻辑;
- 更严格的场景(如金融、图形引擎)可抛出IllegalArgumentException替代自动修正,由调用方决定如何处理错误输入。
通过以上改造,Rect类从一个“裸数据容器”升级为具备内建业务规则的健壮模型——这才是Java面向对象设计的实践精髓。
立即学习“Java免费学习笔记(深入)”;










