有很多的博客都说Java中的基本类型存储在栈上,对象都存储在堆上,这其实是一个非常严重的错误,如果真的是这样,那对象上的基本类型岂不是方法结束之后该变量的值就没有了,事实证明并没有这样。
public class T {
int a = 10000000;
public static void main(String[] args) {
T t = new T();
int c = t.a;
}
}
看这样一个简单的代码,我们通过字节码来看到底是不是存储在栈上的
观察其main方法的字节码指令,非常明显的存在一个指令getfield,这个指令的作用就是从一个类的实例域中获取值,所以说,这个基本类型是与这个对象关联起来的,肯定就是存储在堆上了,所以说,这种言论其实错的非常的明显。
如果我们给a加上final呢?
public class T {
final int a = 10000000;
public static void main(String[] args) {
T t = new T();
int c = t.a;
}
}
可以看到没有了getfield指令了,因为加上了final之后,这个a属性的值肯定就不能改变了,所以java编译器可以字节进行优化,使用这个a属性的值的时候,直接去使用常量池的数据,而不用再getfield了。
并不是说加上了final之后,这个10000000才会被加入常量池,其实加不加final这个10000000都会被加入常量池,他是作为字面量被加入常量池。加上final之后代表了这个a不会被改变,编译器就会进行优化。
还有一点就是,并不是使用了ldc指令之后,a变量在实例中就不占空间了,观察构造函数的字节码我们就能发现其实还是占空间的
这就是上面T的构造函数的字节码,还是会进行初始化的,只是没有使用而已!