0

0

Java继承中变量遮蔽的陷阱与解决方案

霞舞

霞舞

发布时间:2025-08-26 19:44:19

|

974人浏览过

|

来源于php中文网

原创

Java继承中变量遮蔽的陷阱与解决方案

本教程深入探讨Java继承中常见的变量遮蔽(Variable Shadowing)问题。当子类声明与父类同名的实例变量时,可能导致意外的行为,尤其是在多态和条件判断场景下。文章将通过一个具体的开关设备示例,详细解析变量遮蔽的成因、其对程序逻辑的影响,并提供明确的解决方案,指导开发者编写更健壮、可维护的面向对象代码。

深入理解Java中的变量遮蔽

java面向对象编程中,继承是实现代码复用和多态性的核心机制。然而,如果不恰当使用,继承也可能引入一些不易察觉的问题,其中之一便是“变量遮蔽”(variable shadowing)。变量遮蔽发生在子类声明了一个与父类中已存在的实例变量同名且同类型的变量时。此时,子类中的同名变量会“遮蔽”父类中的变量,使得子类实例在访问该变量时,默认访问的是子类自己的变量,而不是父类的变量。这与方法重写(method overriding)不同,方法重写是运行时多态的体现,而变量遮蔽则是一种编译时绑定行为。

案例分析:开关设备的异常行为

考虑一个旨在演示依赖反转原则(DIP)的开关系统。我们有一个抽象基类Switchable,它定义了设备的开关状态state以及turn_on()和turn_off()抽象方法。Lamp和Television是继承自Switchable的具体设备。PowerSwitch类负责通过调用Switchable接口的方法来控制设备。

初始代码结构如下:

// 定义设备状态枚举
public enum State {
    on, off;
}

// 抽象基类:定义可切换设备的通用接口和状态
public abstract class Switchable {
    public State state; // 基类中的状态变量
    abstract public void turn_on(); 
    abstract public void turn_off(); 
}

// Lamp 类:具体设备实现
public class Lamp extends Switchable {
    public State state; // 子类中再次声明了同名状态变量,遮蔽了父类的state
    public Lamp() {
        state = State.off; // 初始化子类自己的state
    }
    public void turn_on() {
        this.state = State.on; // 修改的是子类自己的state
        System.out.println("lamp's on");
    }
    public void turn_off() {
        this.state = State.off; // 修改的是子类自己的state
        System.out.println("lamp's off");
    }
}

// Television 类:另一个具体设备实现
public class Television extends Switchable {
    public State state; // 子类中再次声明了同名状态变量
    public Television() {
        state = State.off;
    }
    public void turn_on() {
        this.state = State.on; 
        System.out.println("lamp's on"); // 注意:这里也错误地打印了"lamp's on"
    }
    public void turn_off() {
        this.state = State.off; 
        System.out.println("lamp's off"); // 注意:这里也错误地打印了"lamp's off"
    }
}

// PowerSwitch 类:负责控制Switchable设备
public class PowerSwitch {
    Switchable sw;  
    public PowerSwitch(Switchable sw) {
        this.sw = sw;
    }
    public void ClickSwitch() {
        // 这里访问的是Switchable引用sw所指向对象中继承自Switchable的state
        if (sw.state == State.off) { 
            sw.turn_on();
        } else {
            sw.turn_off();
        }
    }
}

// 主函数:测试代码
public class Main {
    public static void main(String[] args) {
        Switchable sw = new Lamp(); 
        PowerSwitch ps = new PowerSwitch(sw);
        ps.ClickSwitch(); // 第一次点击
        ps.ClickSwitch(); // 第二次点击
    }
}

在上述代码中,当我们执行Main方法并期望PowerSwitch能正确地切换Lamp的状态时,实际输出却是两次“lamp's off”。这是因为Lamp类内部再次声明了一个public State state;变量,它“遮蔽”了从Switchable基类继承的同名变量。

具体分析如下:

立即学习Java免费学习笔记(深入)”;

ProcessOn
ProcessOn

免费在线流程图思维导图,专业强大的作图工具,支持多人实时在线协作

下载
  1. Switchable基类中的state变量未初始化。 作为一个实例变量,它将默认初始化为null。
  2. PowerSwitch.ClickSwitch()方法中的条件判断: if (sw.state == State.off)。由于sw是一个Switchable类型的引用,它访问的是Lamp对象中继承自Switchable的那个state变量。因为这个变量是null,null == State.off的判断结果是false。
  3. 始终进入else分支: 由于条件判断始终为false,PowerSwitch会一直调用sw.turn_off()。
  4. Lamp.turn_off()方法的行为: Lamp类中的turn_off()方法修改的是this.state,这里的this.state指的是Lamp类自己声明的那个state变量(即被遮蔽的变量),并打印“lamp's off”。

结果是,PowerSwitch所依赖的state(基类变量)始终为null,从未被改变,因此每次都触发turn_off()。而Lamp内部的state(子类变量)虽然被修改,但PowerSwitch无法感知。这种逻辑上的断裂导致了程序行为与预期不符。

解决方案:消除变量遮蔽

解决此问题的关键在于消除子类中的变量遮蔽。一个对象在继承体系中,对于同一个逻辑属性,应该只维护一个实例变量。正确的做法是让Lamp和Television类直接使用从Switchable基类继承的state变量,而不是重新声明一个。

修改后的代码示例如下:

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

826

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

727

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

732

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

445

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

429

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16904

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

177

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.2万人学习

C# 教程
C# 教程

共94课时 | 5.9万人学习

Java 教程
Java 教程

共578课时 | 41.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号