1.动物类(抽象类) abstract class Animal{ private String name; public void setName(String name){ this.name=name; } public String getName(){ return this.name; } 8.抽象类中不一定有抽象方法,抽象方法必须在抽象类中 父类中的抽象方法必须被子类覆盖,否则子类就会继承抽象方法,导致报错 public abstract void doSome();抽象方法! public abstract void eatSome();抽象方法 4.abstract与final无法联合使用 错误:java: 非法的修饰符组合: abstract和final public final abstract void doSome();抽象方法! } 2.猫(非抽象类) class Cat extends Animal{ 8.抽象类中不一定有抽象方法,抽象方法必须在抽象类中 父类中的抽象方法必须被子类覆盖,否则子类就会继承抽象方法,导致报错 注释掉doSome()后会报如下错误 java: Cat不是抽象的, 并且未覆盖Animal中的抽象方法doSome() //public void doSome(){ // System.out.println("猫走猫步"); // } @Override public void eatSome() { System.out.println(super.getName()+"猫吃老鼠"); } public Cat() { } 调用父类的构造方法 public Cat(String name){ super.setName(name); } }
public class Interface { public static void main(String[] args) { ChineseCook c=new ChineseCook(); AmericanCook a=new AmericanCook(); Customer use = new Customer(c,c); c.m=1; //错误:java: 无法为最终变量m分配值 use.getDrink().cha();输出:茶叶泡出的茶 FoodMenu f=new ChineseCook();此处使用了多态 f.shiZiChaoJiDan();输出:东北味的西红柿炒鸡蛋 ChineseCook c2=(ChineseCook)f;强制类型转换后,则可以访问cha()方法 c2.cha();(上一句在被注释后)FoodMenu中没有cha()方法,所以此处会报错 测试没有继承关系的接口之间,进行强制类型转换 编译可以通过,运行时出现错误:ClassCastException 解决方法:加入instanceof判别条件 if(f instanceof Menu) { Menu d = (Menu) f; 如果不加instanceof 判定条件,编译通过,但运行时报错:ClassCastException } 测试:继承与多态同时存在 Cook c3=new Cook("厨师"); c3.cha(); } } 调用接口,此时并不关心实现方法 class Customer{ private FoodMenu food; private DrinkMenu drink; 构造方法 public Customer() { } public Customer(FoodMenu food,DrinkMenu drink) { this.food = food; this.drink = drink; } public DrinkMenu getDrink() { return drink; } public FoodMenu getFood() { return food; } } 接口用来指明方向,交给其他类来具体实现 interface FoodMenu{ void shiZiChaoJiDan();接口中的抽象方法不能有方法体 此处省略了public abstract void yuXiangRouSi();接口中的....不能有方法体 此处省略了public abstract } 接口用来指明方向,交给其他类来具体实现 interface DrinkMenu{ int m=0;此处省略了public static final m只可赋值一次 void cha();此处省略了public abstract } 测试没有继承关系的接口之间,进行强制类型转换 编译可以通过,运行时出现错误:ClassCastException 解决方法:加入instanceof判别条件 interface Menu{ } 实现多个接口 class ChineseCook implements FoodMenu,DrinkMenu{ public void shiZiChaoJiDan(){ System.out.println("东北味的西红柿炒鸡蛋"); } public void yuXiangRouSi(){ System.out.println("四川味的鱼香肉丝"); } public void cha(){ System.out.println("茶叶泡出的茶"); } } 实现多个接口 class AmericanCook implements FoodMenu,DrinkMenu{ public void shiZiChaoJiDan(){ System.out.println("USA的西红柿炒鸡蛋"); } public void yuXiangRouSi(){ System.out.println("USA的鱼香肉丝"); } public void cha(){ System.out.println("coffee粉冲出的coffee"); } } 测试:继承与多态同时存在 class Profession{ String prof; public Profession(String prof) { this.prof = prof; } public Profession() { } } 继承和实现都存在,则extends在前,implements在后 class Cook extends Profession implements FoodMenu,DrinkMenu{ 方法覆盖(实现) public void shiZiChaoJiDan(){ System.out.println(super.prof+"西红柿炒鸡蛋"); } public void yuXiangRouSi(){ System.out.println(super.prof+"鱼香肉丝"); } public void cha(){ System.out.println(super.prof+"coffee"); } 构造方法 public Cook(String prof) { super(prof); } public Cook() { } }
示例程序:
class Animal { public void move(){ System.out.println("动物在移动"); } } class Cat extends Animal { 1.方法覆盖 public void move(){ System.out.println("猫在走猫步"); } 2.子类特有方法,父类要访问需要进行强制类型转换 public void eat(){ System.out.println("猫吃老鼠"); } }
多态:父类型引用指向子类型对象
下图所示例子中,调用的move()方法必须,父子类都共同存在,否则报错.
1.编译阶段(静态绑定):编译器只知道d1是Animal类型,所以在检查语法时,会去Animal.class文件中找d1.move()方法,编译通过,静态绑定成功。
2.运行阶段(动态绑定):在堆内存中创建的Java对象是Cat对象,所以move的时候,真正参与move的对象是一只猫,运行阶段会动态执行Cat对象中的move()方法,运行阶段动态绑定。
3.由本程序可知,move()方法同时存在于父子类中,如若不然会报错。
Animal e1= new Cat(); 1. e1向下转型为e2 Bird e2=(Bird) e1;
Animal e1= new Bird(); 1.instanceof判断条件,防止出现ClassCastException(类型转换错误) if(e1 instanceof Cat) { 2.e1向下转型为e2 Cat e2=(Cat) e1; 3.输出:猫吃老鼠 e2.eat(); } else if(e1 instanceof Bird) { 4.e1向下转型为e2 Bird e2=(Bird) e1; 5.输出:鸟在唱歌 e2.sing(); }