多态性是面向对象编程的又一个重要特征,那么多态是什么呢?

一、多态的概念

1.概念:多态是指在父类中定义的属性和方法被子类继承之后,可以具有不同的数据类型或表现出不同的行为,这使得同一个属性或方法在父类及其各个子类中具有不同的含义。

2.多态现实意义的理解:多态是同一个行为具有多个不同表现形式或形态的能力;多态就是同一个接口,使用不同的实例而执行不同操作。

(1)现实事物经常会体现出多种形态,如学生,学生是人的一种,则一个具体的同学张三既是学生也是人,即出现两种形态。                                                                                                            

(2)Java作为面向对象的语言,同样可以描述一个事物的多种形态。如Student类继承了Person类,一个Student的对象便既是Student,又是Person。

二、多态的使用

1.多态的使用前提:

java实现多态有 3 个必要条件:继承、重写和向上转型

(1)继承:在多态中必须存在有继承关系的子类和父类。

(2)方法重写:子类对父类中某些方法进行重新定义,在调用这些方法时就会调用子类的方法。

(3)向上转型:在多态中需要将子类的引用赋给父类对象,只有这样该引用才既能可以调用父类的方法,又能调用子类的方法。即父类引用变量可以指向子类对象

父类类型 变量名=new 子类类型();

2.示例:接下来我们用代码来帮助理解

父类:Animal类

public class Animal {
	
	public String name;
	public int health;//健康值
	public int love;//亲密度
	
	public Animal() {//无参构造
		super();
	}
	public Animal(String name, int health, int love) {//有参构造
		super();
		this.name = name;
		this.health = health;
		this.love = love;
	}
	
	
	//定义一个方法
	public void test(){
		System.out.println("我是Animal类中的test()方法");
	}
	

}

子类:Cat类

public class Cat extends Animal {

	public String color;//颜色

	public Cat() {
		super();
	}

	public Cat(String name, int health, int love, String color) {
		super(name, health, love);
		this.color = color;
	}


	// 重写Animal类中的test()方法
	public void test() {
		System.out.println("我是Cat类中的test()方法");
	}
	
	public void play(){
		System.out.println("我是Cat类中的play()方法");
	}

}

子类:Dog类

public class Dog extends Animal {
	public String strain;//种类

	public Dog() {
		super();//调用父类Animal类中的无参构造方法
	}

	public Dog(String name, int health, int love, String strain) {
		super(name, health, love);//调用父类Animal类中的有参构造方法
		this.strain = strain;
	}

	
	//重写Animal类中的test()方法
	public void test(){
		System.out.println("我是Dog类中的test()方法");
	}
	
	public void eat(){
		System.out.println("我是Dog类中的eat()方法");
	}
}

测试:Test类

public class Test {

	public static void main(String[] args) {
		
		//创建Dog类对象
		Dog dog = new Dog("旺财", 100, 100, "金毛");
		dog.test();
		
		//创建Cat类对象
		Cat cat = new Cat("Tom", 100, 99, "蓝色");
		cat.test();
		
		System.out.println("--------------------------");
		
		//多态:同一个父类引用,指向不同的子类实例,执行不同的操作。方法重写是实现多态的前提
		
		//向上转型(自动类型转换):父类的引用指向子类的实例(对象)
		Animal animal = new Dog("来福", 100, 98, "泰迪");
		animal.test();
		//eat()方法是Dog类中独有的方法、父类引用无法直接调用子类中特有的方法,如果父类引用需要使用子类中独有的的方法,需要将父类引用强制类型转换为子类
//		animal.eat();
		
		//向下转型(强制类型转换):子类的引用指向父类的引用
		Dog dog2=(Dog)animal;
		dog2.eat();
		
		animal = new Cat("加菲猫", 85, 88, "黄色");
		animal.test();
		//play()方法是Cat类中独有的方法,父类引用无法直接调用子类中特有的方法,如果父类引用需要使用子类中独有的的方法,需要将父类引用强制类型转换为子类
//		animal.play();
		//在向下转型过程中,容易出现类型转换异常ClassCastException,将父类引用转换成了其它的子类对象,所以在转换之前需要对父类引用类型进行判断
//		Dog cat2 =(Dog)animal;
		
		
		if(animal instanceof Dog){
			Dog dog3 =(Dog)animal;
			dog3.eat();
		}else if(animal instanceof Cat){
			Cat cat3 =(Cat)animal;
			cat3.play();
		}
		
		

	}

}

结果:

java 多态 对象数组 java多态的_java 多态 对象数组

由上述代码我们可以看到,创建了一个Animal引用指向Dog类实例,然后使用animal引用调用test()方法,实际上调用的却是Dog类中的test()方法,这就是多态的一种应用;

三、Java中实现和使用多态的主要方式:

1.使用父类作为方法的形参:

示例:

父类:Animal类

public class Animal {
	private String name;
	private int health;//健康值
	private int love;

	public Animal() {
		super();
	}

	public Animal(String name, int health, int love) {
		super();
		this.name = name;
		this.health = health;
		this.love = love;
	}

	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}

	public int getHealth() {
		return health;
	}

	public void setHealth(int health) {
		this.health = health;
	}

	public int getLove() {
		return love;
	}

	public void setLove(int love) {
		this.love = love;
	}

	//定义一个方法实现Animal看病
	public void lookDoctor(){
		System.out.println("我是Animal类中看病的lookDoctor()方法");
	}
}

子类:Cat类

public class Cat extends Animal {
	
	private String color;

	public Cat() {
		super();
	}

	public Cat(String name, int health, int love, String color) {
		super(name, health, love);
		this.color = color;
	}

	public String getColor() {
		return color;
	}

	public void setColor(String color) {
		this.color = color;
	}
	
	//重写Animal类中的lookDoctor()方法
	@Override
	public void lookDoctor() {
		System.out.println("猫生病了,打针和吃药.....");
		this.setHealth(85);
	}

}

子类:Dog类

public class Dog extends Animal {
	
	private String strain;

	public Dog() {
		super();//调用父类Animal类中的无参构造方法
	}

	public Dog(String name, int health, int love, String strain) {
		super(name, health, love);//调用父类Animal类中的有参构造方法
		this.strain = strain;
	}

	public String getStrain() {
		return strain;
	}

	public void setStrain(String strain) {
		this.strain = strain;
	}

	//重写Animal类中的lookDoctor()方法
	@Override
	public void lookDoctor() {
		//健康值小于60的时候
		System.out.println("狗生病了,打针.......");
		this.setHealth(70);
	}
}

子类:Penguin类

public class Penguin extends Animal {

	private char sex;

	public Penguin() {
		super();
	}

	public Penguin(String name, int health, int love, char sex) {
		super(name, health, love);
		this.sex = sex;
	}

	public char getSex() {
		return sex;
	}

	public void setSex(char sex) {
		this.sex = sex;
	}

	// 重写Animal类中的lookDoctor()方法
	@Override
	public void lookDoctor() {
		// 健康值小于60的时候
		System.out.println("企鹅生病了,吃药.......");
		this.setHealth(75);
	}

}

主人类:Master类

public class Master {
	
	//定义给Animal对象看病的方法
	public void cure(Animal animal){
		//animal对象健康值小于60的时候需要看病
		if(animal.getHealth()<60){
			animal.lookDoctor();
		}
		
	}

}

测试类:Test类

public class Test {

	public static void main(String[] args) {
		
		Master master = new Master();
		
		//创建Dog类对象
		Dog dog1 = new Dog("旺财", 30, 99, "藏獒");
		System.out.println(dog1.getHealth());
		
		//调用方法原则一:方法需要什么类型的参数就需要给什么类型的参数---》方法需要什么类型的参数就需要给什么类型的参数(包括其子类)
		master.cure(dog1);
		System.out.println(dog1.getHealth());
		
		System.out.println("----------------");
		
		//创建Penguin类对象
		Penguin penguin1 = new Penguin("QQ", 45, 90, '母');
		System.out.println(penguin1.getHealth());
		
		master.cure(penguin1);
		System.out.println(penguin1.getHealth());
		
		System.out.println("----------------");
		
		Animal animal = new Dog("来福", 20, 100, "拉布拉多");
		System.out.println(animal.getHealth());
		master.cure(animal);
		System.out.println(animal.getHealth());
		
		System.out.println("---------------------------------");
		
		animal = new Penguin("QQ", 50, 92, '公');
		System.out.println(animal.getHealth());
		master.cure(animal);
		System.out.println(animal.getHealth());
	}

}

结果:

java 多态 对象数组 java多态的_数据结构_02

 可以看到在Master类中将Animal类对象作为一个形参,来进行方法的调用;

2.使用父类作为方法的返回值:

父类:Animal类

public abstract class Animal {
	//定义一个动物叫的方法
	public abstract void shout();

}

子类:Cat类

public class Cat extends Animal {

	@Override
	public void shout() {
		System.out.println("喵喵喵");
	}

}

子类:Dog类

public class Dog extends Animal {

	@Override
	public void shout() {
		System.out.println("汪汪汪");
	}

}

主人类:Master类

public class Master {
	//将父类Animal作为方法的形参使用,是多态的使用方式之一
	
	//定义一个方法:实现让动物叫
	public void letShout(Animal animal){
		animal.shout();
	}
	
	//将父类Animal作为方法的返回值,也是多态的使用方式之一
	public Animal giveAnimal(String type){
		Animal animal = null;
		if(type.equals("狗")){
			animal = new Dog();
		}else{
			animal = new Cat();
		}
		
		return animal;
	}

}

测试类:Test类

public class Test {

	public static void main(String[] args) {

		Master master = new Master();

		// Animal是抽象类。不能直接实例化,可以使用多态的形式,将animal引用指向子类实例
		// Animal animal = new Animal();
		Animal animal = new Dog();
		
		master.letShout(animal);
		
		animal = new Cat();
		master.letShout(animal);
		
		System.out.println("-----------------------");
		
		//赠送动物
		Scanner sc = new Scanner(System.in);
		System.out.println("你想要什么动物?(猫/狗)");
		String pet =sc.next();
		Animal ani=master.giveAnimal(pet);
		ani.shout();
	}

}

可以看到Master类中父类Animal类作为方法的返回值。

四、多态的必要性

使用多态的好处:

1.提高了代码的可维护性

2.可以使程序有良好的扩展,并可以对所有类的对象进行通用处理。

五、多态的局限性

1.当父类引用指向子类对象时,父类引用是不能直接调用子类特有的方法的。需要向下转型(强制类型转换)。

向下转型(强制类型转换)格式:

父类类型 父类对象引用=new 子类类型();//向上转型(自动类型转换):父类的引用指向子类的实例(对象

子类类型  子类对象引用=(子类类型)父类对象引用;//向下转型(强制类型转换):子类的引用指向父类的引用

向下转换后才能调用子类特有的方法。

2.同时,在向下转型的过程中,容易出现类型转换异常ClassCastException,将父类引用转换成了其它的子类对象,所以在转换之前需要对父类引用类型进行判断

这时就需要用到instanceof关键字进行判断

instanceof Dog){
             Dog dog =(Dog)animal;
             dog.eat();//Dog类中独有方法
         }else if(animal instanceof Cat){
             Cat cat =(Cat)animal;
             cat.play();//Cat类中独有方法
         }

注意:使用instanceof时,对象的类型必须和instanceof后面的参数所指定的类在继承上有上下级关系