简单来说,Java把内存分为两种:堆内存,栈内存。
栈内存
- 函数(方法)中定义的基本类型的变量和引用变量都在函数的栈内存分配。
当在一段代码块定义一个变量时,Java就在栈中为这个变量分配内存空间,当超过变量的作用域后,Java会自动释放掉为该变量所分配的内存空间,该内存空间可以立即被另作他用。
- 栈内存中的数据可以共享。比如:
int a = 3;
int b = 3;
编译器先处理int a = 3;首先它会在栈中创建一个变量为a的引用,然后查找栈中是否有3这个值,如果没找到,就将3存放进来,然后将a指向3。接着处理int b = 3;在创建完b的引用变量后,因为在栈中已经有3这个值,便将b直接指向3。这样,就出现了a与b同时均指向3的情况。这时,如果再令a=4;那么编译器会重新搜索栈中是否有4值,如果没有,则将4存放进来,并令a指向4;如果已经有了,则直接将a指向这个地址。因此a值的改变不会影响到b的值。
这种情况a的修改并不会影响到b, 它是由编译器完成的,它有利于节省空间
String是一个特殊的包装类数据。可以用:
String str = new String(“abc”);
String str = “abc”;
两种的形式来创建,第一种是用new()来新建对象的,它会在存放于堆中。每调用一次就会创建一个新的对象。
而第二种是先在栈中创建一个对String类的对象引用变量str,然后查找栈中有没有存放"abc",如果没有,则将"abc"存放进栈,并令str指向”abc”,如果已经有”abc” 则直接令str指向“abc”。
即:字符串的字面量是存放在栈中。
堆内存
- 堆内存用来存放由new创建的对象和数组。
- 在堆中分配的内存,由Java虚拟机的自动垃圾回收器来管理。
堆和栈的联系
- 栈内存中引用变量指向堆内存中的对象,存放的是堆内存中对象的地址。
堆和栈的区别
- 栈的存取速度比堆快,仅次于寄存器,栈中的数据大小和生命周期是确定的。
- 堆主要用来存放对象的,栈主要是用来执行程序。
- 栈中确定数据所占空间大小在程序编译期进行,分配是在程序运行时进行;而堆确定和分配都是在程序运行时确定。