为什么父类引用指向子类对象后,子类无法使用之前继承的方法?

我在学习多态的时候遇到一处疑惑的地方,代码如下
public class Test {
    public static void main(String[] args) {
        A c = new C();
        c.commonA;//不报错
        c.commonB;//报错
        c.commonC;//报错
        
    }
}

class A {
    public void commonA(){
        System.out.println("A方法");
    }
}

class B extends A{
    public void commonB(){
        System.out.println("B方法");
    }
}

class C extends B{
    public void commonC(){
        System.out.println("C方法");
    }
}

执行上述代码后,只能正常调用A的方法,而B类和C类的方法均无法调用,这里不是new的C类吗,而C类继承了B类,B类又继承了A类,C类不是就拥有了B类和A类所有的方法吗,那为什么我创建C类的对象后,无法调用他们的方法。
请问这里应该怎么正确理解,这里面的原理又是什么,谢谢

img

1、 A c = new C(); 看上图:虽然C对象里面有commonB()和commonC(),但是C对象的引用你指向了A对象的引用,这在Java语法上是没问题的,子类的对象可以指向父类的引用。可以说是多态的体现。
2、虽然你指向了A对象的引用,由于A对象的引用和C对象的引用在类型上是不一样的。A引用仍然代表A对象。
3、只能访问自己内部的方法,不能访问子类的方法。假如A访问到了子类的方法,那就很可怕了,举个例子:类D继承了A,然后通过特定的方法得到类A,去执行commonB()和commonC(),这就很不安全。假如B对象每天给指定账号打100元,但是D去执行了commonB()两次,结果就是账号收到300.彻底乱套。

不知道你看懂了没?

肯定咯,你new的是C类,但是你是用A来接收的,换句话来说,你new出来的对象被你向上转型成了A。当然就只能用A里的方法了,你要用C继承来的方法,有两种方法:

  1. C c = new C(); // 直接创建C类对象,不对其进行转型
  2. A c = new C(); // 依然用向上转型的方式
    那么你在调用的时候应该是这样,((C) c).commonB(); ((C) c).commonC(); 向下转型为C才可调用C中继承来的方法

这么说吧,如果你把new C()的c看做一个参数传给了一个方法,这个方法是以A 来接受的,如

public void method(A a) {
  ...
}

method(new C());

那是可以正常编译的,因为 C 是 A 的子类,但是在方法 method 中,并不知道 a 是 A 还是 A 的子类 B 或 C,需要判断然后强转才可以执行 A的子类的对应方法,如:

public void method(A a) {
    if (a instanceof B) {
        ((B) a).commonB();
    }
    ...
}

父类引用指向子类对象,就是你爸爸有哪些功能,你就只能使用哪些功能,但是具体的使用是用的你自己的;