Java入门 面向对象(多态、抽象类) (第二十天)

多态的概念:

生活中:同一个事物,作用条件不一样,结果不一样

Java中:同一个父类引用,指向不同的子类实例,执行不同的操作
               方法重写是实现多态的前提   

多态的实现步骤:

多态实现步骤:

    1)在抽象父类中定义抽象方法
    2)子类继承抽象父类并重写父类中所有的抽象方法
    3)测试类中创建父类引用指向不同的子类实例,父类引用调用方法,调用的是子类重写后的那些方法
            向上转型(自动类型转换):父类的引用指向子类的实例
            向下转型(强制类型转换):子类的引用指向父类的引用

父类的引用无法调用子类里独有的方法,必须使用向下转型成对应的子类,才能通过子类去调用子类中独有的方法
    在向下转型的时候,容易出现ClassCastException(类型转换异常),原因是将父类引用转换成了不匹配的子类对象
    可以通过instanceof关键字来判断父类引用指向的是哪一个子类实例,从而避免类型转换异常

使用父类作为方法的形参,是Java中实现和使用多态的主要方式

使用父类作为方法的返回值,也是Java中实现和使用多态的主要方式

抽象: 

抽象方法:使用abstract修饰的方法为抽象方法

          1)抽象方法没有方法体
          2)抽象方法所在的类需要声明为抽象类
          3)子类继承一个抽象类后,子类中必须重写抽象父类中         
             所有的抽象方法,如果
                不重写,那么子类也需要定义为抽象类

抽象类:使用abstract修饰的类为抽象类

      1)抽象类的声明跟之前声明一个类没有太大的区别,依然 
        可以封装属性、声明构造方法、声明getXxx()/setXxx()方 
        法、重写toString()方法等
       2)抽象类中可以包含抽象方法、也可以不包含抽象方法(但
          是抽象方法所在的类一定要声明为抽象类)
       3)抽象类不能实例化(不能通过new的方式来创建对象)

代码(类)

package AnimalDemo04;

public abstract class Father {//使用abstract修饰的类为抽象类
	//定义属性
	private String name;
	private int age;

	//无参构造方法
	public Father() {
		super();
	}
	
	//有参构造方法
	public Father(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}

	
	//get/aet方法
	public String getName() {
		return name;
	}
	public void setName(String name) {
		this.name = name;
	}

	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}

	
     //toString重写方法
	@Override
	public String toString() {
		return "Father [name=" + name + ", age=" + age + "]";
	}
	
	/*public void test(){
		System.out.println("我是Father类中的test()方法");
	}*/
	
	//抽象方法使用abstract修饰,没有方法体
	public abstract void test();

}

子类

package AnimalDemo04;

public class Son1 extends Father {

	private double height;
	
	
	public Son1() {
		super();
	}

	public Son1(String name, int age, double height) {
		super(name, age);
		this.height = height;
	}
	
	
	public double getHeight() {
		return height;
	}
	public void setHeight(double height) {
		this.height = height;
	}

	@Override
	public void test() {
		System.out.println("在Son1类中重写Father类里的test()方法");

	}

}

对象

package AnimalDemo04;

public class Test {
	
	public static void main(String[] args) {

		//创建Father类对象
	    //Father father=new Father();//抽象类不能实例化,因为没有意义
		
		//向上转型:父类引用指向子类的实例
		Father father = new Son1("土灵", 23, 178);
		father.test();	
		
	}
}

转型

向上转型

      语法:
       <父类型> <引用变量名> = new <子类型>();

向上转型:父类的引用指向子类对象,自动进行类型转换

注意:
   1) 此时通过父类引用变量调用的方法是子类 
         覆盖或继承父类的方法,不是父类的方法
   2)此时通过父类引用变量无法调用子类特有 
         的方法

向下转型

 语法:
         <子类型> <引用变量名> = (<子类型> )<父类型的引用变量>;

向下转型:将一个指向子类对象的父类引用赋给一个子类的引用,即:父类类型转换为子类类型。需强制类型转换
   
注意:在向下转型的过程中,如果没有转换为真实子类类型,会出现类型转换异常

代码:父类

package AnimalDemo06;

public abstract class Father {
	
	private String name;
	private int age;


	public Father() {
		super();
	}
	
	public Father(String name, int age) {
		super();
		this.name = name;
		this.age = age;
	}

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

	public int getAge() {
		return age;
	}
	public void setAge(int age) {
		this.age = age;
	}

	
	@Override
	public String toString() {
		return "Father [name=" + name + ", age=" + age + "]";
	}
	
	
	public abstract void test();
	
}

子类:1

package AnimalDemo06;

public class Son1 extends Father {

	private double height;
	
	
	public Son1() {
		super();
	}

	public Son1(String name, int age, double height) {
		super(name, age);
		this.height = height;
	}
	
	
	public double getHeight() {
		return height;
	}
	public void setHeight(double height) {
		this.height = height;
	}

	@Override
	public void test() {
		System.out.println("我是Son1类中重写Father类里的test()方法");

	}
	
	//定义一个Son1类里独有的方法
	public void son1Method(){
		System.out.println("我是Son1类里的独有方法son1Method()");
	}

}

子类:2

package AnimalDemo06;

public class Son2 extends Father {

	private double weight;
	
	
	public Son2() {
		super();
	}
	
	public Son2(String name, int age, double weight) {
		super(name, age);
		this.weight = weight;
	}
	

	public double getWeight() {
		return weight;
	}
	public void setWeight(double weight) {
		this.weight = weight;
	}
	
	@Override
	public void test() {
		System.out.println("我是Son2类中重写Father类里的test()方法");
		
	}
	
	//定义一个Son2类中独有的方法
	
	public void son2Method(){
		System.out.println("我是Son2类中独有的sonMethod()方法");
	}
	

}

 对象(向上转型)

package AnimalDemo06;

public class Test {
	
	public static void main(String[] args) {
		//创建Father类对象
	    //Father father=new Father();//抽象类不能实例化,因为没有意义
		
		//向上转型:父类引用指向子类的实例
		Father father = new Son1("光头强", 23, 178);
		father.test();
	}
}

 对象(向下转型)

package AnimalDemo06;

public class Test {
	
	public static void main(String[] args) {
		//创建Father类对象
	    //Father father=new Father();//抽象类不能实例化,因为没有意义
		

		
		//父类的引用不能调用子类里的方法,需要进行强制类型转换(向下转型)
//		father.son1Method();
		

			//向下转型,子类的引用指向父类的引用
			Son1 son = (Son1)father;
			son.son1Method();

	}
}

instanceof

如何减少在向下转型的过程中,没有转换为真实子类类型的类型转换异常?
        Java中提供了instanceof运算符来进行类型的判断

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

package AnimalDemo06;

public class Test {
	
	public static void main(String[] args) {
		//创建Father类对象
	    //Father father=new Father();//抽象类不能实例化,因为没有意义
		
		//向上转型:父类引用指向子类的实例
		Father father = new Son1("光头强", 23, 178);
		father.test();
		
			
		
		//父类的引用不能调用子类里的方法,需要进行强制类型转换(向下转型)
//		father.son1Method();
		

        //instanceof进行类型的判断
		if(father instanceof Son1){
			//向下转型,子类的引用指向父类的引用
			Son1 son = (Son1)father;
			son.son1Method();
		}else if(father instanceof Son2){
			//向下转型,子类的引用指向父类的引用
			Son2 son= (Son2)father;
			son.son2Method();
		}	
		
	}
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值