所以父类类型的引用可以调用父类中定义的所有属性和方法,对于只存在与子类中的方法和属性它就望尘莫及了。
public class Car {
public void fun1(){
System.out.println("Car 的Fun.....");
fun2();
}
public void fun2(){
System.out.println("Car 的Fun2...");
}
}
public class Truck extends Car{
/**
* @desc 子类重载父类方法
* 父类中不存在该方法,向上转型后,父类是不能引用该方法的
* @param a
* @return void
*/
public void fun1(String a){
System.out.println("Truck 的 Fun1...");
fun2();
}
/**
* 子类重写父类方法
* 指向子类的父类引用调用fun2时,必定是调用该方法
*/
public void fun2(){
System.out.println("Truck 的Fun2...");
}
}
public class Test {
public static void main(String[] args) {
Car a = new Truck();
a.fun1();
}
}
-------------------------------------------------
Output:
Car 的Fun.....
Truck 的Fun2...
public class Car {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Car(){
}
public String getType(){
return "是 " + getName();
}
/**
* 重写toString()
*/
public String toString(){
return null;
}
}
public class Truck extends Car{
public Truck(){
setName("Truck");
}
/**
* 重写父类方法,实现多态
*/
public String getType(){
return "是 " + getName();
}
/**
* 重写toString()
*/
public String toString(){
return "Car : " + getName();
}
}
public class Bus extends Car{
public Bus(){
setName("Bus");
}
/**
* 重写父类方法,实现多态
*/
public String getType(){
return "是 " + getName();
}
/**
* 重写toString()
*/
public String toString(){
return "Car : " + getName();
}
}
public class Test {
public static void main(String[] args) {
//定义父类数组
Car[] car = new Car[2];
//定义两个子类
Truck truck = new Truck();
Bus bus = new Bus();
//父类引用子类对象
car[0] = truck;
car[1] = bus;
for(int i = 0 ; i < 2 ; i++){
System.out.println(car[i].toString() + "--" + car[i].getType());
}
System.out.println("-------------------------------");
}
}
OUTPUT:
Car : Truck--是 Truck
Car : Bus--是 Bus
interface Dao{ //接口的方法全部都是非静态的方法。
public void add();
public void delete();
}
//接口的实现类
class UserDao implements Dao{
public void add(){
System.out.println("添加员工成功!!");
}
public void delete(){
System.out.println("删除员工成功!!");
}
}
class Demo3
{
public static void main(String[] args)
{
//实现关系下的多态
Dao d = new UserDao(); //接口的引用类型变量指向了接口实现类的对象。
d.add();
}
}
经典实例:
public class A {
public String show(D obj) {
return ("A and D");
}
public String show(A obj) {
return ("A and A");
}
}
public class B extends A{
public String show(B obj){
return ("B and B");
}
public String show(A obj){
return ("B and A");
}
}
public class C extends B{
}
public class D extends B{
}
public class Test {
public static void main(String[] args) {
A a1 = new A();
A a2 = new B();
B b = new B();
C c = new C();
D d = new D();
System.out.println("1--" + a1.show(b));
System.out.println("2--" + a1.show(c));
System.out.println("3--" + a1.show(d));
System.out.println("4--" + a2.show(b));
System.out.println("5--" + a2.show(c));
System.out.println("6--" + a2.show(d));
System.out.println("7--" + b.show(b));
System.out.println("8--" + b.show(c));
System.out.println("9--" + b.show(d));
}
}
运行结果:
1--A and A
2--A and A
3--A and D
4--B and A
5--B and A
6--A and D
7--B and B
8--B and B
9--A and D
在这里看结果1、2、3还好理解,从4开始就开始糊涂了,对于4来说为什么输出不是“B and B”呢?
首先我们先看一句话:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。这句话对多态进行了一个概括。其实在继承链中对象方法的调用存在一个优先级:this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。
分析:
从上面的程序中我们可以看出A、B、C、D存在如下关系:
按照同样的方法我也可以确认其他的答案。
方法已经找到了但是我们这里还是存在一点疑问,我们还是来看这句话:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法。这我们用一个例子来说明这句话所代表的含义:a2.show(b);
这里a2是引用变量,为A类型,它引用的是B对象,因此按照上面那句话的意思是说有B来决定调用谁的方法,所以a2.show(b)应该要调用B中的show(B obj),产生的结果应该是“B and B”,但是为什么会与前面的运行结果产生差异呢?这里我们忽略了后面那句话“但是这儿被调用的方法必须是在超类中定义过的”,那么show(B obj)在A类中存在吗?根本就不存在!所以这句话在这里不适用?那么难道是这句话错误了?非也!其实这句话还隐含这这句话:它仍然要按照继承链中调用方法的优先级来确认。所以它才会在A类中找到show(A obj),同时由于B重写了该方法所以才会调用B类中的方法,否则就会调用A类中的方法。
所以多态机制遵循的原则概括为:当超类对象引用变量引用子类对象时,被引用对象的类型而不是引用变量的类型决定了调用谁的成员方法,但是这个被调用的方法必须是在超类中定义过的,也就是说被子类覆盖的方法,但是它仍然要根据继承链中方法调用的优先级来确认方法,该优先级为:this.show(O)、super.show(O)、this.show((super)O)、super.show((super)O)。