深入理解Java的接口和抽象类
抽象类与类的区别
- 抽象类不可以创建实例
- 子类继承抽象类时需要实现抽象方法
- 抽象方法必须是public或者是protected
[public] abstract class ClassName {
abstract void fun();
}
接口
接口是行为的抽象。
[public] interface InterfaceName {
}
类可以实现多个接口
class ClassName implements Interface1,Interface2,[....]{
}
接口中的变量会被隐式地指定为public static final变量(并且只能是public static final变量,用private修饰会报编译错误),而方法会被隐式地指定为public abstract方法且只能是public abstract方法(用其他关键字,比如private、protected、static、 final等修饰会报编译错误),并且接口中所有的方法不能有具体的实现,也就是说,接口中的方法必须都是抽象方法。从这里可以隐约看出接口和抽象类的区别,接口是一种极度抽象的类型,它比抽象类更加“抽象”,并且一般情况下不在接口中定义变量。
接口与抽象类的区别
1.语法层面:
1)抽象类中可以有成员方法的实现,而接口中的成员方法必须是抽象的;
2)抽象类中可以有各种类型的变量,而接口中的对象必须是public static final的对象。
3)抽象类中可以有静态代码块以及静态方法,接口不可以。
4)一个子类只能继承一个抽象类,不过可以实现多个接口。
2.设计层面
1.**抽象类是对一种事物的抽象,即对类抽象,而接口是对行为的抽象。抽象类是对整个类整体进行抽象,包括属性、行为,但是接口却是对类局部(行为)进行抽象。**抽象类是“是不是”,接口是“有没有”,以飞机和鸟举例,飞机和鸟都有飞行的能力,我们可以定义Fly接口包含fly(),然后让具体的类来实现Fly接口。如果一个类继承了某个抽象类,则子类必定是抽象类的种类,而接口实现则是有没有、具备不具备的关系,比如鸟是否能飞(或者是否具备飞行这个特点),能飞行则可以实现这个接口,不能飞行就不实现这个接口。
2.设计层面不同,抽象类作为很多子类的父类,它是一种模板式设计。而接口是一种行为规范,它是一种辐射式设计。模板式设计可以更改公共部分,而子类不需要更改。而辐射式设计更改后,需要将所有使用的都更改。对于抽象类,如果需要添加新的方法,可以直接在抽象类中添加具体的实现,子类可以不进行变更;而对于接口则不行,如果接口进行了变更,则所有实现这个接口的类都必须进行相应的改动。因为如果实现接口,需要把接口中所有的抽象方法都重写一次。
抽象类实现接口与普通类实现接口的区别
通常类实现接口需要实现所有接口,而抽象类不同。抽象类实现接口只需要完成接口中部分方法,而子类继承自抽象类就需要完成抽象类中未实现的所有函数。
举个例子,接口list中包含两个函数size函数与isEmpty函数,而AbstractList1实现list接口,实现了size函数,ArrayList1类继承自AbstractList1类,实现list接口。ArrayList1类就不需要再实现size函数,只需要实现父类未实现的接口中的全部方法。
public class interfaceTry {
public static void main(String[]args){
ArrayList1 list1 = new ArrayList1();
System.out.println(list1.get(0));
System.out.println("size为:"+list1.size());
}
}
interface list{
int size();
boolean isEmpty();
}
abstract class AbstractList1 implements list{
@Override
public int size(){
return 10;
}
abstract public int get(int index);
}
class ArrayList1 extends AbstractList1 implements list{
private int[]sz;
public ArrayList1(){
sz = new int[]{1,2,3,4,5,6,7,8,9};
}
@Override
public int get(int index){
return sz[index];
}
@Override
public boolean isEmpty(){
return true;
}
}