Java继承/实现 extends与implements区别
作者:互联网
extends 是继承父类,只要那个类不是声明final或者定义为abstract就能继承,Java中不支持多重继承,继承只能继承一个类,但implements可以实现多个接口,用逗号分开就行了。
例如:
class A extends B implements C,D,E(){ //class子类名extends父类名implements接口名 }
父类与子类继承关系上的不同:
class A{ int i; void f(){ } } class B extends A{ int j; void f(){} //重写方法 void g(){} }
B b=new B();
b就是子类对象的实例,不仅能够访问自己的属性和方法,也能访问父类的属性和方法。诸如b.i,b.j.b.fn()都是合法的,此时b.f()是B中的f()。
A a=new B();
a虽然是用的B的构造函数,但经过upcast(向上转型),成为父类对象的实例,不能访问子类的属性和方法。a.i,a.f()是合法的,而a.j,a.g()是非法的。此时访问a.f()是访问B中的f()。这里阐述一下具体的实现过程和涉及到的多态。
实现过程:
A a = new B(); 这条语句,实际上有三个过程:
(1) A a;
将a声明为父类对象,只是一个引用,未分配空间。
(2) B temp = new B();
通过B类的构造函数建立了一个B类对象的实例,也就是初始化。
(3) a = (A)temp;
将子类对象temp转换未父类对象并赋给a,这就是向上转型(upcast),是安全的。
经过以上3个过程,a就彻底成为了一个A类的实例。
子类往往比父类有更多的属性和方法,上传只是舍弃,是安全的;而向下转型(downcast)有时会增加方法,通常是不安全的。这里也是instanceof不能判断一个对象是否是真正的父类对象的原因。
多态:
a.f()对应的应该是B类的方法f(),调用构造函数建立实例后,对应方法的入口已经确定了(运行时)。如此一来,a虽被向上转型为A类,但其中重写的方法f()仍然是B的方法f()。也就是说,每个对象知道自己应该调用哪个方法。
A a1 = new B(); A a2 = new C();
a1,a2两个虽然都是A类对象,但各自的f()不同。这正是子类型多态的体现。
接口和抽象类:
- 接口可以继承接口,不能实现接口;抽象类不可以继承接口,但可以实现接口。详见:http://stackoverflow.com/questions/22498245/one-uncertainty-on-multiple-inheritance-in-universal-image-loader
- 抽象类可以继承实体类。抽象类可以实现(implements)接口,抽象类是否可继承实体类,取决于实体类必须是否有明确的构造函数。
- 抽象类可以继承实体类,这是因为抽象类可继承性且有方法。
- 一个接口可以继承多个接口,interface C extends A, B {}是可以的(Java类是单继承的,但接口是可以多继承的)。不允许类多重继承的主要原因是,如果A同时继承B和C,而B和C同时有一个D方法,A如何决定该继承那一个呢?但接口不存在这样的问题,接口全都是抽象方法继承谁都无所谓,所以接口可以继承多个接口。
-
一个类可以实现多个接口: class D implements A,B,C{},但是一个类只能继承一个类,不能继承多个类 class B extends A{}。在继承类的同时,也可以实现多个接口: class E extends D implements A,B,C{} 这也正是选择用接口而不是抽象类的原因。
接口与类的区别:
- 接口不能用于实例化对象。
- 接口没有构造方法。
- 接口中所有的方法必须是抽象方法,Java 8 之后 接口中可以使用 default 关键字修饰的非抽象方法。
- 接口不能包含成员变量,除了 static 和 final 变量。
- 接口不是被类继承了,而是要被类实现。
- 接口支持多继承。
抽象类和接口的区别:
- 抽象类中的方法可以有方法体,就是能实现方法的具体功能,但是接口中的方法不行。
- 抽象类中的成员变量可以是各种类型的,而接口中的成员变量只能是 public static final 类型的。
- 接口中不能含有静态代码块以及静态方法(用 static 修饰的方法),而抽象类是可以有静态代码块和静态方法。
- 一个类只能继承一个抽象类,而一个类却可以实现多个接口。
注:JDK 1.8 以后,接口里可以有静态方法和方法体了。 注:JDK 1.8 以后,接口允许包含具体实现的方法,该方法称为"默认方法",默认方法使用 default 关键字修饰。 注:JDK 1.9 以后,允许将方法定义为 private,使得某些复用的代码不会把方法暴露出去。
举个例子总结一下:
//定义一个接口 public interface Inter { void show(); //抽象方法 default void method() { //默认方法,必须写方法体 System.out.println("默认方法被实现了"); } static void test(){ //静态方法,必须写方法体 System.out.println("静态方法被实现了"); } } //定义接口的一个实现类 public class Interlmpl implements Inter { @Override public void show() { System.out.println("show方法"); } } //定义测试类 public class InterDemo { public static void main(String[] args) { Inter i = new Interlmpl(); i.show(); //抽象方法强制被重写 i.method(); //默认方法不强制被重写,但可以被重写,重写时去掉default关键字 Inter.test(); //静态方法只能通过接口名调用,不能通过实现类名或者对象名调用 } }
2022-06-14 20:50:57
标签:implements,Java,继承,接口,extends,抽象类,方法,class 来源: https://www.cnblogs.com/gb-0526/p/16376328.html