变量遮蔽指子类或方法内同名变量覆盖父类/成员变量,需用this访问当前对象成员、super访问父类成员;构造器中this()和super()须首行且互斥;遮蔽是静态绑定,不同于方法重写的动态绑定。

当子类中定义了与父类同名的变量,或者方法内部定义了与成员变量同名的参数/局部变量时,就发生了“变量遮蔽”(Variable Shadowing)。此时若想明确访问被遮蔽的成员变量,需借助 this(当前对象的成员)或 super(父类继承来的成员)关键字。
成员变量被形参遮蔽时用 this
构造器或方法中,常将参数名设为与成员变量一致(如 name),便于理解但会遮蔽成员变量。不加 this 会导致赋值操作作用于参数自身,而非对象字段。
class Person {
private String name;
public Person(String name) {
name = name; // ❌ 错误:形参赋给自己,成员变量未被初始化
}
}
正确写法是用 this.name 明确指向当前对象的成员变量:
-
this.name = name;—— 将形参值赋给当前对象的name字段 -
this只能用于非静态上下文(实例方法、构造器),静态方法中不可用 - 即使没有遮蔽,加
this也能提升可读性,尤其在复杂类中
子类变量遮蔽父类同名变量时用 super
若子类声明了与父类同名的实例变量(不推荐但合法),则子类对象中实际存在两个同名变量:一个来自父类(可通过 super 访问),一个属于子类自身(默认访问)。
立即学习“Java免费学习笔记(深入)”;
class Animal { String kind = "animal"; }
class Dog extends Animal {
String kind = "dog"; // 遮蔽父类 kind
void show() {
System.out.println(kind); // 输出 "dog"
System.out.println(this.kind); // 输出 "dog"(等价)
System.out.println(super.kind); // 输出 "animal"
}
}
-
super.kind强制访问父类中被遮蔽的变量,仅适用于继承关系中的实例变量 -
super不能用于静态变量(应直接用父类名.静态变量) - 遮蔽变量易引发混淆,建议通过命名区分(如父类用
baseKind,子类用specificKind)
this 与 super 在构造器中的特殊规则
构造器中若需调用本类其他构造器或父类构造器,this(...) 或 super(...) 必须作为第一条语句出现,且二者不可共存。
-
this(...)调用本类重载构造器,用于代码复用(如统一初始化逻辑) -
super(...)调用父类构造器,确保父类部分被正确初始化 - 若未显式写出
super(),编译器自动插入无参super();若父类无无参构造器,必须显式调用super(...)
遮蔽 vs 重写:别混淆变量和方法
变量遮蔽(field shadowing)和方法重写(method overriding)机制不同:前者是静态绑定(编译期决定访问哪个变量),后者是动态绑定(运行期根据实际类型调用)。
- 多态下,调用的是子类重写的方法,但访问的是引用类型声明处的变量(不是实际对象类型)
- 例如:
Animal a = new Dog(); a.kind取的是Animal.kind,而a.speak()执行的是Dog.speak() - 因此,避免在子类中遮蔽父类变量,更不要依赖遮蔽实现“多态变量”










