文章目录
12.抽象类和接口
抽象方法
-
使用 abstract 修饰的方法,没有方法体,只有声明。
-
定义的是一种“规范”,就是告诉子类必须要给抽象方法提供具体的实现。
1.抽象类
- 包含抽象方法的类就是抽象类。
- 通过抽象类,我们就可以做到严格限制子类的设计,使子类之间更加通用。
抽象类的使用要点:
- 有抽象方法的类只能定义成抽象类
- 抽象类不能实例化,即不能用 new 来实例化抽象类。
- 抽象类可以包含属性、方法、构造方法。但是构造方法不能用来 new 实例,只能用来被子类调用。
- 抽象类只能用来被继承。
- 抽象方法必须被子类实现。
抽象类和抽象方法的基本用法
package opp.b;
/**
* 测试抽象类
* 1. 有抽象方法的类只能定义成抽象类
* 2. 抽象类不能实例化,即不能用 new 来实例化抽象类。
* 3. 抽象类可以包含属性、方法、构造方法。但是构造方法不能用来 new 实例,
* 只能用来被子类调用。
* 4. 抽象类只能用来被继承。
* 5. 抽象方法必须被子类实现。
*/
abstract class Animal {
int age;
public abstract void rest();
public void shout(){
System.out.println("Animal.shout");
// Animal a=new Animal(); 2
}
}
class Dog extends Animal {
//子类必须实现父类的抽象方法,否则编译错误 5
@Override
public void rest() {
}
}
public class TestAbstractClass{
public static void main(String[] args) {
Dog a=new Dog();
a.shout();
}
}
2.接口(interface)
接口就是一组规范(就像我们人间的法律一样),所有实现类都要遵守。
-
为什么需要接口?接口和抽象类的区别?
接口就是比“抽象类”还“抽象”的“抽象类”,可以更加规范的对子类进行约束。全面地专业地实现了:规范和具体实现的分离。
接口是两个模块之间通信的标准,通信的规范。如果能把你要设计的模块之间的接口定义好,就相当于完成了系统的设计大纲,剩下的就是添砖加瓦的具体实现了。大家在工作以后,做系统时往往就是使用“面向接口”的思想来设计系统。
接口和实现类不是父子关系,是实现规则的关系。比如:我定义一个接口 Runnable,Car 实现它就能在地上跑,Train 实现它也能在地上跑,飞机实现它也能在地上跑。就是说,如果它是交通工具,就一定能跑,但是一定要实现 Runnable 接口。
a.如何定义和使用接口
声明格式:
[访问修饰符] interface 接口名 [extends 父接口 1,父接口 2…] {
常量定义;
方法定义;
}
定义接口的详细说明:
访问修饰符:只能是 public 或默认。
接口名:和类名采用相同命名机制。
*extends:*接口可以多继承。
常量:接口中的属性只能是常量,总是:public static final 修饰。不写也是。
方法:接口中的方法只能是:public abstract。 省略的话,也是 public abstract。
要点
子类通过 implements 来实现接口中的规范。
接口不能创建实例,但是可用于声明引用变量类型。
一个类实现了接口,必须实现接口中所有的方法,并且这些方法只能是 public 的。
JDK1.8(不含 8)之前,接口中只能包含静态常量、抽象方法,不能有普通属性、构造方法、普通方法。
JDK1.8(含 8)后,接口中包含普通的静态方法、默认方法。
package opp.testinterface;
//飞行接口
public interface Volant {
/*public static final*/ int Fly_HIGHT=100;
/*public abstract*/ void fly();
}
//善良接口
interface Honest{
void helpPther();
}
class Goodman implements Honest{
@Override
public void helpPther() {
System.out.println("扶老奶奶过马路!");
}
}
class BirdMan implements Volant{
@Override
public void fly() {
System.out.println("我在飞!");
}
}
class Angel implements Volant,Honest{
@Override
public void fly() {
System.out.println("Angel.fly");
}
@Override
public void helpPther() {
System.out.println("Angel.helpOther");
}
}
class Plant implements Volant{
@Override
public void fly() {
System.out.println("飞机.在飞");
}
}
package opp.testinterface;
public class Test {
public static void main(String[] args) {
Angel a=new Angel();
a.fly();
a.helpPther();
System.out.println(Volant.Fly_HIGHT);
Volant a2=new Angel();
a2.fly();
// 不能用a2.helpOther(); //Volant类型的对象中没有这个方法,需转型
}
}
输出结果:
b.接口中定义静态方法和默认方法(JDK8)
JAVA8 之前,接口里的方法要求全部是抽象方法。
JAVA8(含 8)之后,以后允许在接口里定义默认方法和静态方法。
JDK8 新特性_默认方法
Java 8 及以上新版本,允许给接口添加一个非抽象的方法实现,只需要使用 default 关键字即可,这个特征又叫做默认方法(也称为扩展方法)。
默认方法和抽象方法的区别是抽象方法必须要被实现,默认方法不是。作为替代方式,
接口可以提供默认方法的实现,所有这个接口的实现类都可以得到默认方法。
package opp.testinterface;
//测试接口中的新特性(默认方法)
public interface TestDefault {
void printfo();
default void moren(){
System.out.println("TestDefault.moren");
System.out.println("测试默认方法");
}
}
class TestDefaultImp01 implements TestDefault{
@Override
public void printfo() {
System.out.println("TestDefaultImp01.printInfo");
}
}
package opp.testinterface;
public class Test {
public static void main(String[] args) {
System.out.println("=================测试默认方法================");
TestDefault td=new TestDefaultImp01();
td.printfo();
td.moren();
}
}
输出结果:
JDK8 新特性_静态方法
JAVA8 以后,我们也可以在接口中直接定义静态方法的实现。这个静态方法直接从属于接口(接口也是类,一种特殊的类),可以通过接口名调用。
如果接口的实现类中定义了相同名字的静态方法,那就是完全不同的方法了,直接从属于实现类。可以通过实现类名直接调用。
package opp.testinterface;
/**
* 测试静态方法
* 接口的默认方法中可以调用静态方法,反过来不行。
*/
public class test01 {
public static void main(String[] args) {
A a = new Test_A();
a.moren();
} }
interface A {
public static void staticMethod(){
System.out.println("A.staticMethod");
}
public default void moren(){
staticMethod();
System.out.println("A.moren");
} }
class Test_A implements A {
public static void staticMethod(){
System.out.println("Test_A.staticMethod");
}
}
c.接口的多继承
接口支持多继承。和类的继承类似,子接口 extends 父接口,会获得父接口中的一切。
package opp.testinterface;
/**
* 测试接口的多继承
*/
public class TestMultipleInheritance {
public static void main(String[] args) {
C c=new CImpl01();
c.testE();
c.testC();
c.testB();
}
}
interface E{
void testE();
}
interface B{
void testB();
}
interface C extends E,B{
void testC();
}
class CImpl01 implements C{
@Override
public void testB() {
System.out.println("CImpl01.testB");
}
@Override
public void testC() {
System.out.println("CImpl01.testC");
}
@Override
public void testE() {
System.out.println("CImpl01.testE");
}
}