马士兵教程里的一个问题

interface Singer {
public void sing();
public void sleep();
}

interface Painter {
public void paint();
public void eat();
}

class Student implements Singer {
private String name;
Student(String name) {
this.name = name;
}

public void sing() {
System.out.println("student is singing");
}

public void sleep() {
System.out.println("student is sleeping");
}
}

class Teacher implements Singer,Painter {
private String name;
Teacher(String name) {
this.name = name;
}
public void sing() { System.out.println("teacher is singing"); }
public void sleep() { System.out.println("teacher is sleeping"); }
public void paint() { System.out.println("teacher is painting"); }
public void eat() { System.out.println("teacher is eating"); }
}

public class Test {
public static void main(String args[]) {
Singer s1 = new Student("chenfei");
s1.sing();s1.sleep();s1.study();
Singer s2 = new Teacher("xiong");
s2.sing();s2.sleep();
Painter p1 = (Painter)s2;
p1.paint();p1.eat();
}
}

这个程序里 Painter p1 = (Painter)s2; 这句话将s2这个Teacher对象强制转换成painter类,那么p1就指向一个painter类对象了,就不存在基类引用指向子类对象了,为什么下面的p1.paint();这个方法还能执行出teacher is painting呢??

对于函数覆盖来说,执行的是这个对象实际类型(用new创建的那个对象类型)拥有的方法,而不是你引用类型的方法。
这个你应该从面向对象的设计的角度去思考,而不是纠结在语法和具体编译器实现的层面。
派生类覆盖基类的函数,这个语法的实际用途是允许对类库调用的外围程序员改变基类代码的行为。而编写类库的程序员肯定不知道外围程序员所定义的类型
所以不管是什么语言,C++ Java Python还是什么,都必须是实际类型的方法覆盖引用类型的方法,这样的编程语言才有实用的用途。
好比不管是电动车、混动车、汽油车、柴油车,都要具有踩下油门踏板让车往前行驶的功能。你用不着去理解汽油机、柴油机、电动机的原理,你买车就是开的。
同样编程语言就是用来写程序的。