一文详解Java中的多态

1 多态和类型转换

多态指的是同一个方法调用,由于对象不同可能会有不同的行为。现实生活中,同一个方法,具体实现会完全不同。 比如:同样是调用人的“休息”方法,张三是睡觉,李四是旅游,高淇老师是敲代码,数学教授是做数学题; 同样是调用人“吃饭”的方法,中国人用筷子吃饭,英国人用刀叉吃饭,印度人用手吃饭。

多态的要点:

  1. 多态是方法的多态,不是属性的多态(多态与属性无关)。
  2. 多态的存在要有 3 个必要条件:继承方法重写父类引用指向子类对象
  3. 父类引用指向子类对象后,用该父类引用调用子类重写的方法,此时多态就出现了。
class Animal {
	public void shout() {
		System.out.println("叫了一声!");
	}
}
class Dog extends Animal {
	public void shout() {
		System.out.println("旺旺旺!");
	}
	public void seeDoor() {
		System.out.println("看门中....");
	}
}
class Cat extends Animal {
	public void shout() {
		System.out.println("喵喵喵喵!");
	}
}
public class TestPolym {
	public static void main(String[ ] args) {
		Animal a1 = new Cat(); // 向上可以自动转型
		//传的具体是哪一个类就调用哪一个类的方法。大大提高了程序的可扩展性。
		animalCry(a1);
		Animal a2 = new Dog();
		animalCry(a2);//a2 为编译类型,Dog 对象才是运行时类型。
		/*编写程序时,如果想调用运行时类型的方法,只能进行强制类型转换。
		* 否则通不过编译器的检查。*/
		Dog dog = (Dog)a2;//向下需要强制类型转换
		dog.seeDoor();
	}
	// 有了多态,只需要让增加的这个类继承 Animal 类就可以了。
	static void animalCry(Animal a) {
		a.shout();
	}
	/* 如果没有多态,我们这里需要写很多重载的方法。
	* 每增加一种动物,就需要重载一种动物的喊叫方法。非常麻烦。
	static void animalCry(Dog d) {
		d.shout();
	}
	static void animalCry(Cat c) {
		c.shout();
	}*/
}

父类引用做方法的形参,实参可以是任意的子类对象,可以通过不同的子类对象实现不同的行为方式。
多态也有弊端,就是无法调用子类特有的功能,比如,我不能使用父类的引用变量调用 Dog类特有的 seeDoor()方法。

2 对象的转型(casting)

父类引用指向子类对象,我们称这个过程为向上转型,属于自动类型转换。
向上转型后的父类引用变量只能调用它编译类型的方法,不能调用它运行时类型的方法。这时,我们就需要进行类型的强制转换,我们称之为向下转型!

public class TestCasting {
	public static void main(String[ ] args) {
		Object obj = new String("北京尚学堂"); // 向上可以自动转型
		// obj.charAt(0) 无法调用。编译器认为 obj 是 Object 类型而不是 String 类型
		/* 编写程序时,如果想调用运行时类型的方法,只能进行强制类型转换。
		* 不然通不过编译器的检查。 */
		String str = (String) obj; // 向下转型
		System.out.println(str.charAt(0)); // 位于 0 索引位置的字符
		System.out.println(obj == str); // true.他们俩运行时是同一个对象
	}
}

在向下转型过程中,必须将引用变量转成真实的子类类型(运行时类型)否则会出现类型转换异常 ClassCastException。如下代码所示。

public class TestCasting2 {
	public static void main(String[ ] args) {
		Object obj = new String("北京尚学堂");
		//真实的子类类型是 String,但是此处向下转型为 StringBuffer
		StringBuffer str = (StringBuffer) obj;
		System.out.println(str.charAt(0));
	}
}

为了避免出现这种异常,我们可以使用 instanceof 运算符进行判断。

public class TestCasting3 {
	public static void main(String[ ] args) {
		Object obj = new String("北京尚学堂");
		if(obj instanceof String){
			String str = (String)obj;
			System.out.println(str.charAt(0));
		}else if(obj instanceof StringBuffer){
			StringBuffer str = (StringBuffer) obj;
			System.out.println(str.charAt(0));
		}
	}
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值