java_抽象类、接口、多态的使用
抽象类
抽象类:包含抽象方法的类。
抽象方法 : 没有方法体的方法。
使用 abstract
关键字修饰方法,该方法就成了抽象方法,抽象方法只包含一个方法名,而没有方法体。
格式为:修饰符 abstract 返回值类型 方法名 (参数列表);
抽象类的使用
继承抽象类的子类必须重写父类所有的抽象方法。否则,该子类也必须声明为抽象类。最终,必须有子类实现该父类的抽象方法,否则,从最初的父类到最终的子类都不能创建对象,没有意义。
public class Demo1 {
public static void main(String[] args) {
//抽象类不能创建对象
// Employee e=new Employee();
Manager m=new Manager();
m.work(); //管理其他人
Cooker c=new Cooker();
c.work(); //炒菜
}
}
//员工类
abstract class Employee{
private String id;
private String name;
private Double salary;
public Employee() {
}
public Employee(String id, String name, Double salary) {
this.id = id;
this.name = name;
this.salary = salary;
}
//抽象方法不能有方法体
public abstract void work();
}
class Manager extends Employee{
//必须重写父类的抽象方法
@Override
public void work() {
System.out.println("管理其他人");
}
}
class Cooker extends Employee{
//必须重写父类的抽象方法
@Override
public void work() {
System.out.println("炒菜");
}
}
注意事项
- 抽象类不能创建对象,如果创建,编译无法通过而报错。只能创建其非抽象子类的对象。
- 抽象类中,可以有构造方法,是供子类创建对象时,初始化父类成员使用的。
- 抽象类中,不一定包含抽象方法,但是有抽象方法的类必定是抽象类。
- 抽象类的子类,必须重写抽象父类中所有的抽象方法,否则,编译无法通过而报错。除非该子类也是抽象类。
接口
接口,是Java语言中一种引用类型,接口的内部封装了方法,包含抽象方法(JDK 7及以前),默认方法和静态方法(JDK 8),私有方法 (JDK 9)。
引用数据类型:数组,类,接口。
接口的定义,它与定义类方式相似,但是使用 interface 关键字
接口的使用,它不能创建对象,但是可以被实现( implements ,类似于被继承) 。一个实现接口的类(可以看做是接口的子类),需要实现接口中所有的抽象方法,创建该类对象,就可以调用方法了,否则它必须是一个抽象类。
格式:
public interface 接口名称 {
// 抽象方法
// 默认方法
// 静态方法
// 私有方法
}
含有抽象方法:使用 abstract 关键字修饰,可以省略,没有方法体。该方法供子类实现使用。
public interface InterFaceName {
public abstract void method();
}
含有默认方法和静态方法(JDK1.8新特性)
默认方法:使用 default 修饰,不可省略,供子类调用或者子类重写。
静态方法:使用 static 修饰,供接口直接调用。
public interface InterFaceName {
public default void method() {
// 执行语句
}
public static void method2() {
// 执行语句
}
}
含有私有方法和私有静态方法(JDK1.9新特性)
私有方法:使用 private 修饰,供接口中的默认方法或者静态方法调用。
public interface InterFaceName {
private void method(){
}
}
基本的实现
类与接口的关系为实现关系,即类实现接口,该类可以称为接口的实现类,也可以称为接口的子类。实现的动作类 似继承,格式相仿,只是关键字不同,实现使用 implements 关键字。
非抽象子类实现接口:
- 必须重写接口中所有抽象方法。
- 继承了接口的默认方法,即可以直接调用,也可以重写。
抽象方法的使用:必须全部实现
定义接口:
public interface LiveAble {
// 定义抽象方法
public abstract void eat();
public abstract void sleep();
}
定义实现类:
public class Animal implements LiveAble {
@Override
public void eat() {
System.out.println("吃饭");
}
@Override
public void sleep() {
System.out.println("睡觉");
}
}
测试类:
public static void main(String[] args) {
Animal a=new Animal();
a.eat();
a.sleep();
}
输出结果:
吃饭
睡觉
接口的多实现
class 类名 [extends 父类名] implements 接口名1,接口名2,接口名3... {
// 重写接口中抽象方法【必须】
// 重写接口中默认方法【不重名时可选】
}
接口中,有多个抽象方法时,实现类必须重写所有抽象方法。如果抽象方法有重名的,只需要重写一
次。
接口的多继承
一个接口能继承另一个或者多个接口,这和类之间的继承比较相似。接口的继承使用 extends 关键字,子接口继 承父接口的方法。如果父接口中的默认方法有重名的,那么子接口需要重写一次。
其他成员特点
接口中,无法定义成员变量,但是可以定义常量,其值不可以改变,默认使用public static final修饰。
接口中,没有构造方法,不能创建对象。
接口中,没有静态代码块。
接口中成员变量默认加上public static final修饰
interface A{
int num=1;
//相当于:public static final int num=1;
}
接口中的普通成员方法默认被public abstract修饰
interface A{
void method();
//相当于:public abstract void method();
}
多态
多态: 是指同一行为,具有多个不同表现形式。
多态体现的格式:
父类类型 变量名 = new 子类对象;
变量名.方法名();
父类类型:指子类对象继承的父类类型,或者实现的父接口类型。
当使用多态方式调用方法时,首先检查父类中是否有该方法,如果没有,则编译错误;如果有,执行的是子类重写 后方法。
定义父类:
abstract class Animal{
public abstract void eat();
}
定义子类:
class Cat extends Animal{
@Override
public void eat(){
System.out.println("吃鱼");
}
}
class Dog extends Animal{
@Override
public void eat() {
System.out.println("吃骨头");
}
}
测试类:
public class Demo1 {
public static void main(String[] args) {
//多态形式,创建对象
Animal cat=new Cat();
//调用的是Cat的eat
cat.eat();
//多态形式,创建对象
Animal dog=new Dog();
//调用的是Dog的eat
dog.eat();
}
}
引用类型转换
多态的转型分为向上转型与向下转型两种:
向上转型:多态本身是子类类型向父类类型向上转换的过程,这个过程是默认的。当父类引用指向一个子类对象时,便是向上转型。
父类类型 变量名 = new 子类类型();
如:Animal a = new Cat();
向下转型:父类类型向子类类型向下转换的过程,这个过程是强制的。一个已经向上转型的子类对
象,将父类引用转为子类引用,可以使用强制类型转换的格式,便是向下转型。
子类类型 变量名 = (子类类型) 父类变量名;
如:Cat c =(Cat) a;
转型的异常
ClassCastException ,类型转换异常
为了避免 ClassCastException 的发生,Java提供了 instanceof 关键字,给引用变量做类型的校验
变量名 instanceof 数据类型
如果变量属于该数据类型,返回true。
如果变量不属于该数据类型,返回false。