欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

JAVA中的父类私有成员变量的继承问题

程序员文章站 2022-07-15 16:50:08
...

如果父类中属性为私有(private),那么能否被子类继承呢?
答案是不可以。
我们看如下简单代码

class Father {
    private String name;
    
    public void sayHi() {
        System.out.println("My name is " + this.name);
    }
}

class Son extends Father {}

public class PrivateFieldTest {
    public static void main(String[] args) {
        Father f1 = new Father();
        Son s1 = new Son();
		f1.sayHi();
		s1.sayHi();
    }
}

得到的结果是:

My name is null
My name is null

这里我们使用的都是默认构造函数,子类自动引用父类的默认构造函数。直接构造为null。

这样,Son类继承了Father类的sayHi方法,那么自然,Son的sayHi方法中使用的“name”变量,自然就是Son内部继承自Father的私有变量name了,也就是说私有变量可以被继承?

不是的。

我们再看下一段代码,我们在Son的类中重写一下sayHi方法。

class Father {
    private String name;
    
    public void sayHi() {
        System.out.println("My name is " + this.name);
    }
}

class Son extends Father {
    public void sayHi() {
        System.out.println("My name is " + this.name);
    }
}

public class PrivateFieldTest {
    public static void main(String[] args) {
        Father f1 = new Father();
        Son s1 = new Son();
        f1.sayHi();
        s1.sayHi();
    }
}

这里,出现了编译错误,即Son类里面并没有继承name.
JAVA中的父类私有成员变量的继承问题
我们看看错误的原因。

The field Father.name is not visible

可以看见,编译器自动认为,name是属于Father的,Son内并没有继承。
那为什么第一段代码中,可以使用sayHi方法得到数据呢?

实际上,这样解释比较好:

“子类不能继承父类的私有属性,但如果子类中公有的方法影响到了父类的私有属性,那么私有属性是能够被子类使用的。”

这句话听起来很拗口,但是实际情况确实也很拗口。
看如下代码

class Father {
    private String name;
    
    public void setName(String name) {
        this.name = name;
    }
    public void sayHi() {
        System.out.println("My name is " + name);
    }
}

class Son extends Father {}

public class PrivateFieldTest {
    public static void main(String[] args) {
        Father f1 = new Father();
        Son s1 = new Son();
        f1.sayHi();
        s1.sayHi();
        System.out.println();
        f1.setName("Sam");
        f1.sayHi();
        s1.sayHi();
        System.out.println();
        s1.setName("Tom");
        f1.sayHi();
        s1.sayHi();
    }
}

运行结果是

My name is null
My name is null

My name is Sam
My name is null

My name is Sam
My name is Tom

第一段结果,没有变化。

第二段结果,对应的是我们使用setName方法改变了f1的name,所以f1对应的sayHi结果变成了Sam。

第三段结果,对应的是我们使用setName方法改变了s1的name,所以s1对应的sayHi结果变成了Tom。

由此我们可以看到,虽然子类不能继承父类私有变量,但是还是可以通过公有方法使用私有变量。只是重写函数的时候可能比较麻烦,所以要活用super。

以上为个人理解,有错欢迎随时探讨~