今天有同学拿着这样一段代码来问我这个是不是java接口的实例化,代码如下:
class MyThread implements Runnable{..... }
//另一个类的main方法中:
Runnable r = new MyThread();
我就告诉他这个其实很明显,是父类声明指向子类的引用。具体实例化的是MyThread这个对象。然后进行了思考,针对如下代码:
Thread t = new Thread(new Runnable(){
@Override
public void run(){.... }
})
这段代码大家也许并不陌生,那么有人要问了,既然接口不能实例化,Runnable是一个接口,此处不是用new来实例化了么。这个其实就是一个假象的实例化而已,这种叫做匿名内部类,实质的代码等同于如下代码:
class MyThread implements Runnable{
@Override
public viod run(){ ........... }
}
//另一个类的main方法中
Thread t = new Thread(new MyThread());
只是上面的代码省略掉了类的名字而已。
java的接口为什么不能实例化呢?
首先,我们需要明白实例化的含义。实例化实际意义是在jvm的堆中开辟出一块内存空间,比如Student s = new Student();此处声明Student对象s,并且实例化一个Student对象,实则是在堆中开辟出一块空间来存放Student对象,s则是指向这块空间,也就是内存中的一块地址。这块地址中所存放的值就是我们这个Student对象的一些属性。
具体类所占用的内存空间(也就是堆里的一块地址)所存放的值是类的成员变量,这里插入一张图便于我们理解:
图中可以看到,栈中存放的是per,值是堆中具体Person这个对象的地址,也就是per指向这个具体类的引用。而堆中一块地址,存放的值是Person这个类的成员变量(局部变量在执行时存放在栈中)。
那么换做一个接口呢?假设我们可以new出一个接口,那么这个接口就会占用堆中的一块地址,那么我们想想接口的这块地址上能够存放什么值呢。
说到这个首先要清楚接口中允许有什么:静态的属性以及方法声明。这里再说明下,java的内存空间分为四类:栈(stack)、堆(heap)、代码(code)、静态数据(data)。由此可见接口中所有的东西的具体值都是存放在代码区和静态数据区的,所以接口的这块地址上并没有任何实际的值需要存储,那么为什么要给他一块地址来浪费空间呢。其实用底层代码在内存中开辟出一块空间很容易,那么为什么java设定不允许接口实例化呢,以我的理解而言,那就是接口的实例化没有任何实际意义,只会占用一块内存空间,却不会在这块空间中放任何实际的值,所以java主动去规避掉了这个问题。
原文:https://www.cnblogs.com/ltycomeon/p/13562245.html