Java中的栈和堆的区别是什么
栈和堆的区别是什么?
为什么说栈的速度快,堆的速度慢?
JVM在处理他们的时候,有什么区别么
------解决方案--------------------java 的堆是一个运行时数据区,类的(对象从中分配空间。这些对象通过new、newarray等指令建 立,它们不需要程序代码来显式的释放。堆是由垃圾回收来负责的,堆的优势是可以动态地分配内存大小,生存期也不必事先告诉编译器,因为它是在运行时动态分 配内存的,Java的垃圾收集器会自动收走这些不再使用的数据。但缺点是,由于要在运行时动态分配内存,存取速度较慢。
栈的优势是,存取速度比堆要快,仅次于寄存器,栈数据可以共享。但缺点是,存在栈中的数据大小与生存期必须是确定的,缺乏灵活性。栈中主要存放一些基本 类型的变量(,int, short, long, byte, float, double, boolean, char)和对象句柄。
基本数据类型存储在“栈”中,对象引用类型实际存储在“堆”中,在栈中只是保留了引用内存的地址值。
------解决方案--------------------
栈中的数据占内存大小在编译时是确定的,比如一个int类型就占4B,所以变量地址好计算,所以分配和销毁和访问速度都比较快.
堆中的数据占内存大小一般在编译时是不确定的,在运行时才能知道大小,所以其地址只有在运行时计算,而且运行时可能占内存大小还有变动,所以对这样的数据的分配,销毁和访问都非常不方便,速度也慢一些.
------解决方案--------------------我自己的总结:
栈里放的是地址,堆里可以放数据也可以放地址(想象下堆里的东西也有可能指向别的地方)
每个地址都会指向给定的数据,不然就没有存在的必要了,同样的道理,堆中的数据没有被指针指向的话,也没有存在的必要了,所以当obj=null时就释放内存了。
Java有个好处就是没有指针,Java中的传递的都是传引用,不像c++还能传地址,比如指针p++和p+1两个的结果完全不同。
------解决方案--------------------
栈中的数据占内存大小在编译时是确定的,比如一个int类型就占4B,所以变量地址好计算,所以分配和销毁和访问速度都比较快.
堆中的数据占内存大小一般在编译时是不确定的,在运行时才能知道大小,所以其地址只有在运行时计算,而且运行时可能占内存大小还有变动,所以对这样的数据的分配,销毁和访问都非常不方便,速度也慢一些.
------解决方案--------------------
Java虚拟机提供了程序运行时环境,运行时环境中最重要的一个资源是运行时数据区。
运行时数据区是操作系统为Java虚拟机进程分配的内存区域,并由虚拟机管辖,同时这块区域又分为若干个子区域,
主要包括堆区、方法区和Java栈区。
在堆区中存放对象,
在方法区存放类的类型信息,类型信息包括静态变量和方法信息,方法信息中包含类的所有方法的字节码。
栈,有个特点,先进去的后出来,里边放的是进程中正在执行的方法,
比如说方法
public void A(){
B();
}
public void B(){
C();
}
public void C(){
}
假设一个程序只调用A()方法,(开始时栈是空的):
程序运行的时候,先执行A()方法,因为A方法调用了B()方法,这时因为A()没有执行完,中断A()方法,将其放入栈中暂存,(此时因为栈原来是空的,A方法被push进栈后,位于栈底),执行B()方法,执行时因为B方法调用了C()方法,此时中断B方法的执行,将B()方法push入栈,由于栈中已经有A方法,B被push进去后,位于A的上面;C()方法执行完后,去栈中查找位于栈顶的方法,查到是方法B,方法B继续执行,执行完后再去栈中查找位于栈顶的方法,此时栈内只又一个方法,那就是方法A,方法A有继续执行,A执行完后,栈中已经没有方法,程序运行结束。
假设下边这个柱状的东西是栈
-----------------------------------
| |
|B()|---位于栈口最近的方法先被pop出来执行完成
|A()|---栈底的方法最后执行完
———
------解决方案--------------------栈是编译时分配空间,而堆是动态分配(运行时分配空间),所以栈的速度快
------解决方案--------------------cpu有专门的寄存器(esp,ebp)来操作栈,堆都是使用间接寻址的。栈快点。