主要内容:
•抽象类
•接口
•instanceof运算符
•abstract可以修饰类和方法:
1、abstract修饰的类称为做抽象类;
2、abstract修饰的方法叫做抽象方法,抽象方法只有声明部分,而没有具体的方法体。
一个abstract类只关心它的子类是否具有某种功能,并不关心其自身功能的具体行为,功能的具体行为由子类负责实现。
抽象类不能被直接实例化,即不能使用关键字new来创建该抽象类的对象。
由于Mammal类为抽象类,所以不能直接使用new关键字创建该类的对象
•抽象类中可以没有abstract方法(为了强迫使用者必须通过继承来使用这个类);但是一旦类中包含了abstract方法,则这个“类”一定是abstract类,即有抽象方法的类一定是抽象类。
•抽象类的子类必须实现抽象类中的所有抽象方法,否则子类也必须是抽象类。
•抽象类中的抽象方法是多态的一种表现形式。
•抽象类与普通类区别:
1、抽象类前面由abstract修饰,而普通类没有。
2、抽象类不能创建对象,普通类可以创建对象。
3、抽象类中可以有抽象方法,普通类中一定没有抽象方法
思考:
abstract final class Mammal{ } 能编译通过吗, why?
原因:如果抽象类前面可以添加final就意味着该类无法被继承,也就意味着该抽象类中的抽象方法永远无法得到实现,也就意味着抽象类中的抽象方法是无用的。
Mammal抽象类中move抽象方法的访问权限可以为private吗,即“private abstract void move();”, why?
原因:被private修饰的方法其作用范围为本类,如果抽象类中的抽象方法被private修饰就意味着该方法永远无法被实现。
Mammal抽象类中move抽象方法可以由static修饰吗,,即“public static abstract void move();” why?
原因:抽象类中的抽象方法如果可以被static修饰就意味着可以使用抽象类的类名来使用该方法,但是该抽象方法没有方法体,不具有使用的价值,所以Java中规定抽象类中不能包含被static修饰的静态抽象方法
•接口:
•Java接口是抽象方法的集合,其定义语法为:
访问权限控制符 interface 接口名 [extends 接口列表] {
常量;
抽象方法;
内部类;
}
1、接口名的命名规则和类名的命名规则相同
2、该访问控制符只有两个:public和友好的
•接口内只能包含常量(public static final)、抽象方法及内部类。
•接口中的抽象方法必须为public访问权限控制符,不能为其它控制符。
•通过extends关键字可以使自定义的接口实现继承,但需要注意以下两点:
接口只能继承父接口,不能继承抽象类和普通类。
接口弥补了Java单一继承的缺点(Java中的类只能继承一个父类),即接口可以继承多个父接口,它们之间用英文逗号隔开。
接口实现:
•类通过implements关键字实现接口,Java中的类只能是单继承,但却可以实现多个接口以弥补Java类单继承的不足,其语法如下:
访问控制符 修饰符 class 类名 implements 接口1 [,接口2, ……] {
变量;
方法;
}
•注意:在类中实现接口时,实现类中方法的名字、返回值类型、参数的个数及参数数据类型必须与接口中的对应的抽象方法完全一致。
•如果一个类实现了一个接口,但没有实现接口中的所有抽象方法,那么这个类必须是abstract类。
•如果多个接口中定义了相同的抽象方法,则在实现类中只实现其中一个即可;
接口回调:•接口回调描述的是一种现象:接口声明的变量指向其实现类实例化的对象,那么该接口变量就可以调用接口中的抽象方法。
•接口没有构造方法,不能创建自己的对象,但是可以引用实现类的对象。
•接口实现类可以直接使用接口中的常量:
•接口实现类所实现的多个接口中有常量名相同的常量,则在实现类中不能直接使用,必须使用类名来确定到底调用哪个接口中的常量:
•Java8以前版本中规定,接口中所定义的方法只能是抽象方法,从Java8开始,接口中可以添加一个或多个由default关键字修饰的非抽象方法,该方法又称为扩展方法,该默认方法将由接口实现类创建的对象来调用,如下例:
public interface IMammal {
void move();
public default void eat() {
System.out.println("哺乳动物正在吃......");
}
}
•同样,从Java8开始,接口中可以添加一个或多个由static关键字修饰的非抽象方法,该方法将由接口或其实现类直接调用,如下例:
interface IMammal {
void move();
static void eat() {
System.out.println("哺乳动物正在吃......");
}
}
函数式接口:
•如果接口内只定义一个抽象方法,则该接口称为函数式接口,如下例:
@FunctionalInterface
interface IMammal {
String NAME = "哺乳动物";
void move();
public default void eat() {
System.out.println("哺乳动物正在吃......");
}
}
•注意:
Ø可以使用@FunctionalInterface 注解来验证一个接口是不是函数式接口,Java8中java.lang.Runnable、java.util.Comparator<T>都是函数式接口;
Ø函数式接口中可以定义多个常量、多个默认方法和多个静态方法,但只能定义一个抽象方法及多个java.lang.Object中的public方法,如下例:
@FunctionalInterface
interface IMammal {
void move();
boolean equals(Object obj);
}
抽象类与接口的比较:
| 抽象类 | 接口 |
关键字 | abstract | Interface |
成员变量 | 可包含任意合法成员变量(包括各种访问级别的类成员变量和实例成员变量) | 只能包含公开静态常量(默认由public static final修饰) |
构造方法 | 有构造方法 | 无构造方法 |
方法 | 可包含任意合法方法(包括各种访问级别的非抽象类方法和非抽象实例方法,也包含除private外的非静态抽象方法) | JDK7及其以下版本JDK只能包含公开且抽象的方法(默认由public abstract修饰),而JDK8开始可以包含default修饰的非抽象方法。 |
如何实现抽象方法 | 通过自定义类继承抽象类的方式实现抽象类的抽象方法 | 通过自定义类implements接口实现接口中的抽象方法,定义类可以implements多个接口 |
是否存在多继承 | 一个抽象类只能继承一个抽象或非抽象类 | 一个接口可以继承多个接口 |
•instanceof运算符 :
•instanceof运算符用于判断该运算符前面引用类型变量指向的对象是否是后面类,或者其子类、接口实现类创建的对象。如果是则返回true,否则返回false,其使用格式如下:
引用类型变量 instanceof (类、抽象类或接口)
判断前面对象是否是后面类创建的对象
判断前面对象是否是后面类子类创建的对象
•instanceof运算符用于强制类型转换之前检查对象的真实类型以避免类型转换异常,从而提高代码健壮性。
•代码分析:无法保证传入print方法的实参一定为Date对象,如果传入的不是由Date或其子类创建的对象,那么上面红框框起的代码就会引发“java.lang.ClassCastException”异常,如何解决这个问题呢?