多态 指的是对象的多种形态
继承是多态的实现基础,别忘了子父类要有继承关系.
多态特性:
一、引用多态
1.父类引用可以指向本类对象 Animal obj1 = new Animal();
2.父类引用可以指向子类对象 Animal obj2 = new Dog();
但是我们不能用子类的引用指向父类对象
Dog obj3 = new Animal();//错
二、方法多态
创建本类对象时,调用的方法为本类方法。
创建子类对象时,调用的方法为子类重写的方法或者继承的方法。
1.在父类Animal中定义一个eat()方法,输出一个语句(动物有吃的能力); 在子类Dog中重写eat()方法,输出一个语句(狗是吃肉的); 那么我们在测试类main函数里面,如果obj1.eat() ,那么调用的是父类的方法. 若用obj2调用eat()方法,那么调用的是子类的方法.
2.还有一种情况,比如创建一个继承父类Animal的子类Cat ,但是Cat里并不重写继承的eat()方法. 然后,我们在测试类main函数里创建一个子类对象, Animal obj3 = new Cat(); 然后调用 obj3.eat(); 那么,结果调用的则是子类继承父类的方法. (输出结果:动物有吃的能力)
3.最后一种特殊情况,多态的特性是不能使用的. 若在子类添加一个独有的方法 public void watchDoor() ,含有一句输出语句(狗具有看门的能力); 那么我们在测试类的main函数当中(得先定义好对象Animal obj2 = new Dog() ),就不能用obj2.watchDoor(),即不能通过父类的引用调用子类的方法
1 public class Animal { 2 public void eat(){ 3 System.out.println("动物有吃的能力"); 4 } 5 } 6 7 public class Dog extends Animal { 8 public void eat(){ 9 System.out.println("狗是吃肉的"); 10 } 11 public static void main(String[] args) { 12 Animal obj1 = new Animal(); 13 Animal obj2 = new Dog(); 14 obj1.eat(); 15 obj2.eat(); 16 } 17 }
引用类型转换:
1.向上类型转换(隐式/自动类型转换),是小类型到大类型的转换。 如:Dog dog=new Dog();Animal animal=dog;//正确,自动类型提升,向上类型转换
2.向下类型转换(强制类型转换),是大类型到小类型的转换(存在风险,溢出) 如:Dog dog1=(Dog)animal;//向下类型转换
3.instanceof运算符,来解决引用对象的类型,避免类型转换的安全性问题。
如: Dog dog=new Dog();
Animal animal=dog;
Cat cat=(Cat)animal;//编译时不会出错(按Cat类型进行编译),但运行时会报错,因为它开辟的是Dog类型的空间,而(无法将引用类型进行转换)无法将dog对象转换成Cat类型,并且此方法对程序的安全性有影响。
此时应该利用instanceof和if语句结合使用,进行验证,以保证程序的安全性,如:
if(animal instanceof Cat){
//判断animal类中是否包含Cat类型的元素,若包含则进行转换,instanceof返回值为布尔类型
1 if (dog instanceof cat){ 2 Dog l1 = (Dog)cat; 3 }else{ 4 System.out.println("无法进行类型转换!"); 5 }
抽象类
1 语法定义:
类前使用abstract关键字修饰,则该类为抽象类
2 应用场景:
a.在某些情况下,某个父类只是知道其子类应该包含怎样的方法, 但无法准确知道这些子类如何实现这些方法。抽象类约束子类必须有这些方法,而并不关注子类如何实现。
b.从多个具有相同特征的类中抽象出一个抽象类,以这个抽象类作为子类的模板,从而避免了子类设计的随意性。
3 作用:
限制规定了子类必须实现某些方法,但不关注实现细节
4使用规则:
a.abstract定义抽象类
b.abstract定义抽象方法,只有声明,不需要具体实现。
c.包含抽象方法的是抽象类(如果在类前不适用abstract关键字,会出错)
d.抽象类中可以包含普通的方法,也可以没有抽象方法。
e.抽象类不能直接创建对象,可以定义引用变量,指向一个子类的对象
接口:
1.接口可以理解为一种特殊的类,由全局常量和公共抽象方法组成
如果说类是一种具体实现体,而接口定义了某一批类所需要遵守的规范,接口不关心这些类的内部数据,也不关心这些类里方法的具体实现细节,它只规定这些类里必须提供某些方法
2.接口的定义:
和类定义不同,定义接口不再使用class关键字,而是使用interface关键字.
基本语法:[修饰符] interface 接口名[extends 父接口1,父接口2.......]{
零个到多个常量定义...
零个到多个抽象方法定义...
}
接口就是用来继承,被实现,修饰符一般建议用public
注意:不能够使用private和protected修饰接口
3 接口定义:常量
接口中的属性是常量,即使定义时不添加public static final修饰符,系统也会自动加上
方法:
接口中的方法只能是抽象方法,即使定义时不添加public abstract字符修饰,系统也会自动加上
4实现:
一个类可以实现一个或多个接口,实现接口使用implements关键字。
java中一个类只能继承一个父类,是不够灵活的,通过实现多个接口可以做补充,java中一个类可以实现一个或者多个接口。
继承父类实现接口的语法:
[修饰符] class 类名 extends 父类 implements接口1 ,接口2......{
类体部分//如果继承了抽象类,需要实现继承的抽象方法
//要实现接口中的抽象方法
}
如果要继承父类,继承父类必须在实现接口之前
5 实现时可以利用接口的引用指向实现了接口的对象,调用其方法,如 :
IPlay ip1=new Psp();ip1.playGame();
Telphone抽象类只提供打电话,发短信的方法,没有提供玩游戏的功能,怎么在不改变抽象方法的情况下,让智能手机拥有玩游戏的功能?这就用到了我们的接口
我们想描述Psp呢?Psp只能玩游戏不能发短信和打电话
1 public interface IplayGame { 2 public void playGame(); 3 } 4 5 public abstract class Telphone { 6 public abstract void call(); 7 public abstract void message(); 8 } 9 10 public class SmartPhone extends Telphone implements IplayGame { 11 12 @Override 13 public void call() { 14 // TODO Auto-generated method stub 15 System.out.println("智能手机能打电话!"); 16 } 17 18 @Override 19 public void message() { 20 // TODO Auto-generated method stub 21 System.out.println("智能手机能够发短信!"); 22 } 23 24 @Override 25 public void playGame() { 26 // TODO Auto-generated method stub 27 System.out.println("智能手机能够玩游戏!"); 28 } 29 } 30 31 public class Psp implements IplayGame{ 32 @Override 33 public void playGame() { 34 // TODO Auto-generated method stub 35 System.out.println("psp能够玩游戏!"); 36 } 37 } 38 39 public class Cellphone extends Telphone { 40 41 @Override 42 public void call() { 43 // TODO Auto-generated method stub 44 System.out.println("普通手机能够打电话!"); 45 } 46 47 @Override 48 public void message() { 49 // TODO Auto-generated method stub 50 System.out.println("普通手机能够发短信!"); 51 } 52 } 53 54 public class Ceshi { 55 public static void main(String[] args) { 56 IplayGame smp = new SmartPhone(); 57 smp.playGame(); 58 Telphone t1 = new SmartPhone(); 59 t1.call(); 60 t1.message(); 61 IplayGame ip = new Psp(); 62 ip.playGame(); 63 } 64 }
6接口在使用过程中,还经常与匿名内部类配合使用
匿名内部类就是没有名字的内部类,多用于关注实现而不关注实现类的名字
public class Nmnbl { IplayGame ip1 = new IplayGame() { public void playGame() { System.out.println("我是匿名内部类实现接口!"); }; }; public static void main(String[] args) { new IplayGame() { @Override public void playGame() { // TODO Auto-generated method stub System.out.println("使用第二中匿名内部类实现接口!"); } }.playGame(); } }
我们在命名一个接口时,在首字母加I来区分普通类和接口