任何语言所编写的程序,其中的各类型的数据都需要一个存储位置,java中书的存储位置分为以下5种:
1、寄存器
最快的存储区,位于处理器内部,但是数量及其有限。所以寄存器根据需求自动分配,无序人为控制。
2、栈内存
位于RAM中,通过堆栈指针可以从处理器中获得直接支持。堆栈指针向下移动,则分配新的内存;向上移动,则释放哪些内存。这种存储方式仅次于寄存器。(常用于存放对象引用和基本数据类型,而不用于存储对象)
3、堆
一种通用的内存池,也位于RAM中。其中存放的数据由JVM自动进行管理。
堆相对于栈的好处来说:编译器不需要知道存储的数据在堆里存活多长。当需要一个对象时,使用new写一行代码,当执行这行代码时,会自动在堆里进行存储分配。同时,因为以上原因,用堆进行数据的存储分配和清理,需要花费更多的时间。
4、常量池
常量(字符串常量和基本类型常量)直接存储在程序内部(常量池)。这样做事安全的,因为它们的值在初始化时就已经被确定,并不会改变,常量池在java泳衣保存在编译已确定的,已编译的class文件中的一份数据。它包括了关于类、方法、接口中的常量,也包括字符串常量,如String s="java"这种生命方式。
5、非RAM存储区
如果数据完全存活于程序之外,那么它可以不受到任何的程序控制,在程序没有运行时也可以存在。其中的两个基本的例子是:流对象和持久化对象
java中数据的存储分为以上5种方式,但是在实际中最常谈起的是:堆内存存储和栈内存存储。
我们可以联系二者来分析这两种不同的存储方式,更利于我们理解:
首先,它们有一定的相同之处:
堆和栈都是用于程序中的数据在RAM(内存)上的存储区域。而且java会自动的管理堆和栈,不需要人为去设置
其次,更关键的在于它们的不同之处:
1、存储数据类型:栈内存中存放局部变量(基本数据类型和对象引用),而堆内存用于存放对象(实体)
2、存储速度:就存储速度而言,堆内存的存储分配和清理速度更快与堆,并且栈内存的存储速度仅次于直接位于处理器中的寄存器
3、灵活性:就灵活性而言,由于栈内存和堆内存的存储机制的不同,堆内存灵活性更优于栈内存。
这两种存储方式的不同之处,也是由于它们自身的存储机制所造成的。所以为了理解它们,首先我们应该弄清楚它们分别的存储原理和机制,在java中:
--栈内存被要求存放在其中的数据的大小,生命周期必须是已经确定的;
--堆内存可以被虚拟机动态的分配内存大小,无序事先告诉编译器的数据的大小、生命周期等相关信息。
接下来便可以进行分析:
栈内存和堆内存的存储数据类型为何不同?
我们知道在java中,变量的类型通常分为:基本数据类型和对象引用变量。
首先,8中基本数据类型中的数字类型实际上都是存储的一组位数(所占bit位)不同的二进制数据;除此之外,布尔型只有true和false两种可能值。
其次,对象引用变量存储,实际上是其所关联(指向)对象在内存中的内存地址,而内存地址实际上也是一串二进制的数据。
所以,局部变量的大小是可以被确定的;
接下来,java中局部变量会在其自身属性方法(或代码块)执行完毕后,被自动释放。
所以局部变量的生命周期也是可以被确定的。
那么,既然局部变量的大小和生命周期都可以被确定,完全符合我们栈内存的存储特点。自然,局部变量被存放在栈内存中
---------------------------------------------------------------------------------------------------------------------------
而java中使用关键字new通过调用类的构造函数,从而得到该类的对象
对象类型数据在程序编译期,并不会在内存中创建和存储工作;而是在程序运行期,才根据需要进行动态的创建和存储
也就是说,在程序运行之前,我们永远不能确定这个对象的内容、大小、声明周期。自然,对象由堆内存进行存储管理。
为什么栈内存的速度高于堆内存?
1、栈中数据一般是基本数据类型和对象引用,实际值基本都是一串二进制数据,数据比较小。而堆数据存放的对象类型数据一般更大。
2、局部变量大小、生命周期确定,堆中的对象不能确定。
3、由于2的原因,栈中的对象可以更方便地进行回收,也就是清理及时。
为什么堆的灵活性要强于栈?
这个比较好理解,就是堆中的数据不确定生命周期和大小所以灵活啊
一个很重要的特点:
栈中存储的数据可以存在数据共享。也就是多个栈中的引用可以指向同一个堆对象
一点补充:
对象 对象名=new 对象。
这种方式实际上是编译阶段执行=前面部分,就是声明对象,然后运行阶段执行后面部分,也就是实例化。String s="ab";和String s=new String("ab");区别就在于此。