在抽象类中,可以包含一个或多个抽象方法;但在接口(interface)中,所有的方法必须都是抽象的,不能有方法体,它比抽象类更加“抽象”。
接口(interface)是我们开发java项目,必须用到的方法,而接口是一种完全抽象的设计,没有任何实现。
接口(interface)的特征:
1.所有的成员变量都是public、static、final类型。
2.所有的方法都是public、abstract类型。
3.所有的嵌套类型(类或接口)都是public、static类型。
由此可知,接口中的所有成员都是public的,因为接口是抽象的,必须由其他类所实现,故成员一定要具备足够的访问权限。
如果声明一个接口,并且在接口中没有显式声明任何成员,那个这个接口为空吗?在这个问题之前,先看一个简单的实例。
例:
package deep;
public class UpCast implements Graph {
@Override
public void draw() { }public void cast() { }
public static void main(String[] args) {
UpCast u = new UpCast();
u.draw();
u.cast();
Graph g = u;
g.draw();
// g.cast(); 不能调用cast方法
}
}interface Graph {
void draw();
}
当类上转为接口时,只能调用接口中声明的方法,而不能够调用自己类中声明的方法。根据这个例子可以说明,通过接口类型的引用,所能调用的方法应该是在接口中声明的,否则无法调用。
于是,我们来测试下没有声明任何成员的接口,看看其是否为空。
例:
package deep;
public class EmptyInterfaceTest implements EmptyInterface {
public static void main(String[] args) throws InterruptedException {
EmptyInterface e = new EmptyInterfaceTest();
e.equals(null);
e.getClass();
e.hashCode();
e.notify();
e.notifyAll();
e.toString();
e.wait();
e.wait(100L);
e.wait(100L, 100);
}
}interface EmptyInterface {
}
这个程序可以通过编译。也许大家会感觉很奇怪,在EmptyInterface接口中没有声明任何成员,为什么通过接口引用(e)还可以调用这一系列方法呢?这是因为,接口从来都不是空的,即使我们没有显式声明任何方法,在接口中也默认存在9个方法,这9个方法与Object类中声明的9个public方法相对应。这也说明,即使没有显式声明任何成员的“空接口”,事实上也并非为空,因为至少存在9个方法成员.
接口不能实例化
接口是一种完全抽象的设计,不能实例化,即我们不能创建接口类型的对象,因为这样的对象没有任何实现,是毫无意义的。可是,下面的程序怎么来解释呢?
例:
package deep;
public class Instantiated {
public static void main(String[] args) {
Bird b = new Bird() {
@Override
public void fly() {
System.out.println(“flying”);
}
};
b.fly();
}
}interface Bird {
void fly();
}
虽然不能像创建对象那样使用new来实例化接口,也似乎使用new Bird来创建了接口的实例,并且实现了fly方法。该程序可以通过编译,运行结果如下:
flying
一切都运转正常,这是否说是接口也可以实例化呢?
其实,这一切都是假象而已。接口是完全抽象的设计,不可以实例化。细心的读者也会发现,在编译Instantiated.java后,会生成3个class文件,分别为Instantiated.class、Bird.class、Instantiated$1.class,前两个并不奇怪,关键是第3个class文件从何而来呢?
程序中的创建方式,是使用匿名类来实现的,第3个class文件也就是从这里产生的。其实,程序中的new Bird并没有真正的创建一个Bird类型的“接口对象”,而是创建了一个匿名类。该类实现了Bird接口,并且实现了Bird接口中的fly方法。
口的继承
在Java中,类不允许多重继承,因为从多个类继承的成员及其容易造成混淆与错用,Java中去除了这一特性。不过,对于接口来说,因为其设计是完全抽象的,不包含任何实现,因而接口的继承与类的继承相比,问题相对较少,故Java中接口是可以多重继承的。
接口中的方法都是abstract类型的,目的是要求实现接口的类去实现这些抽象方法,并通过接口引用来指向实现接口的对象,这样就可以调用接口中的方法。故接口中都是实例方法,不允许声明static(静态)方法。因为静态方法是不依赖对象而存在的,可以通过类名直接调用,也不需要创建对象。
如果子接口声明了与父接口中相同名称的变量,就会隐藏父类接口中的同名变量。
如果两个接口中声明了相同名称的变量,当一个类实现了这两个接口,或者子接口多重继承这两个接口,则对该同名变量访问的时候,必须使用限定名称,使用简单名称就会引发编译错误。
例:
package deep;
interface Donkey {
String kind = “donkey”;
}interface Horse {
String kind = “horse”;
}interface Mule extends Donkey, Horse {
// String des = kind; The field kind is ambiguous
String des2 = Donkey.kind;
String des3 = Horse.kind;
}class MuleClass implements Mule {
// String des = kind; The field kind is ambiguous
String des2 = Donkey.kind;
String des3 = Horse.kind;
}class MuleClass2 implements Donkey, Horse {
// String des = kind; The field kind is ambiguous
String des2 = Donkey.kind;
String des3 = Horse.kind;
}
接口作为类型使用
接口作为引用类型来使用,任何实现该接口的类的实例都可以存储在该接口类型的变量中,通过这些变量可以访问类中所实现的接口中的方法,Java 运行时系统会动态地确定应该使用哪个类中的方法,实际上是调用相应的实现类的方法。