Java对象的上转型对象与下转型

将一个类型强制转换成另一个类型的过程被称为类型转换。本节所说的对象类型转换,是指存在继承关系的对象,不是任意类型的对象。当对不存在继承关系的对象进行强制类型转换时,会抛出 Java 强制类型转换(java.lang.ClassCastException)异常。

假设,A类是B类的父类,当用子类创建一个对象,并把这个对象的引用放到父类的对象中时,比如:

A  a;
a = new B();
或
A a;
B  b = new B();
a = b;

 上转型对象

这时,称对象a是对象b的上转型对象.

1.上转型对象不能操作子类新增的成员变量(失掉了这部分属性);不能调用子类新增的方法(失掉了一些功能)。

2.上转型对象可以访问子类继承或隐藏的成员变量,也可以调用子类继承的方法或子类的重写方法。上转型对象操作子类继承的方法或子类重写的方法,其作用等价于子类对象去调用这些方法。因此,如果子类重写了父类的某个方法后,当对象的上转型对象调用这个方法时一定是调用了子类重写的方法。

①不要将父类创建的对象和子类对象的上转型对象混淆。

②可以将对象的上转型对象再强制转换到一个子类对象,这时,该子类对象又具备了子类所有属性和功能。

③不可以将父类创建的对象的引用赋值给子类声明的对象 

class Anthropoid 
{
	double m = 12.58;
	void crySpeak(String s){
		System.out.println(s);
	}
}
class People extends Anthropoid
{
	char m = 'A';
	int n = 60;
	void computer(int a, int b){
		int c = a + b;
		System.out.println(a + "加" + b + "等于" + c);
	}
	void crySpeak(String s){
		System.out.println(super.m + "**" + s + "**" + m);
	}
}
class Example5_9 
{
	public static void main(String[] args) 
	{
		People people = new People();
		Anthropoid monkey = people;   //monkey是people的上转型对象
		monkey.crySpeak("I LOVE THIS GAME");  //等同于people调用重写的crySPeak方法
		//monkey.n = 100;   非法,n是子类新增的成员变量
		//monkey.computer(12,19);   非法,子类新增的方法
		System.out.println(monkey.m);  //操作隐藏的m,不等同于people.m
		System.out.println(people.m);  //操作子类的m
		//把上转型对象强制转换为子类对象
		People zhang = (People)monkey;
		zhang.computer(55, 33);
		zhang.m = 'Z';
		System.out.println(zhang.m);
	}
}

public class Animal {
    public String name = "Animal:动物";
    public static String staticName = "Animal:可爱的动物";

    public void eat() {
        System.out.println("Animal:吃饭");
    }

    public static void staticEat() {
        System.out.println("Animal:动物在吃饭");
    }
}
public class Cat extends Animal {
    public String name = "Cat:猫";
    public String str = "Cat:可爱的小猫";
    public static String staticName = "Dog:我是喵星人";

    public void eat() {
        System.out.println("Cat:吃饭");
    }

    public static void staticEat() {
        System.out.println("Cat:猫在吃饭");
    }

    public void eatMethod() {
        System.out.println("Cat:猫喜欢吃鱼");
    }

    public static void main(String[] args) {
        Animal animal = new Cat();
        Cat cat = (Cat) animal; // 向下转型
        System.out.println(animal.name); // 输出Animal类的name变量
        System.out.println(animal.staticName); // 输出Animal类的staticName变量
        animal.eat(); // 输出Cat类的eat()方法
        animal.staticEat(); // 输出Animal类的staticEat()方法
        System.out.println(cat.str); // 调用Cat类的str变量
        cat.eatMethod(); // 调用Cat类的eatMethod()方法
    }
}

通过引用类型变量来访问所引用对象的属性和方法时,Java 虚拟机将采用以下绑定规则:

  • 实例方法与引用变量实际引用的对象的方法进行绑定,这种绑定属于动态绑定,因为是在运行时由 Java 虚拟机动态决定的。例如,animal.eat() 是将 eat() 方法与 Cat 类绑定。
  • 静态方法与引用变量所声明的类型的方法绑定,这种绑定属于静态绑定,因为是在编译阶段已经做了绑定。例如,animal.staticEat() 是将 staticEat() 方法与 Animal 类进行绑定。
  • 成员变量(包括静态变量和实例变量)与引用变量所声明的类型的成员变量绑定,这种绑定属于静态绑定,因为在编译阶段已经做了绑定。例如,animal.name 和 animal.staticName 都是与 Animal 类进行绑定。

强制对象类型转换

Java 编译器允许在具有直接或间接继承关系的类之间进行类型转换。对于向下转型,必须进行强制类型转换;对于向上转型,不必使用强制类型转换。

例如,对于一个引用类型的变量,Java 编译器按照它声明的类型来处理。如果使用 animal 调用 str 和 eatMethod() 方法将会出错,如下:

animal.str = "";    // 编译出错,提示Animal类中没有str属性
animal.eatMethod();    // 编译出错,提示Animal类中没有eatMethod()方法

如果要访问 Cat 类的成员,必须通过强制类型转换,如下:

((Cat)animal).str = "";    // 编译成功
((Cat)animal).eatMethod();    // 编译成功

把 Animal 对象类型强制转换为 Cat 对象类型,这时上面两句编译成功。对于如下语句,由于使用了强制类型转换,所以也会编译成功,例如:

Cat cat = (Cat)animal;    // 编译成功,将Animal对象类型强制转换为Cat对象类型

类型强制转换时想运行成功就必须保证父类引用指向的对象一定是该子类对象,最好使用 instanceof 运算符判断后,再强转,例如:

Animal animal = new Cat();
if (animal instanceof Cat) {
    Cat cat = (Cat) animal; // 向下转型
    ...
}

猜你喜欢

转载自blog.csdn.net/qq_43629083/article/details/108690133