1.内存分配的区域
(1) 寄存器:这是最快的保存区域,这是主要由于它位于处理器内部。然而,寄存器的数量十分有限,所以寄存器是需要由编译器分配的。我们对此没有直接的控制权,也不可能在自己的程序里找到寄存器存在的任何踪迹。
(2) 堆栈:位于通用RAM(随机访问存储器)中。可通过它的“堆栈指针” 获得处理的直接支持。堆栈指针若向下移,会创建新的内存;若向上移,则会释放那些内存。这是一种特别快、特别有效的数据保存方式,仅次于寄存器。Java编译器必须准确地知道堆栈内保存的所有数据的“长度”以及“存在时间”
(3) 堆:在RAM区域,用来保存Java对象。编译器不必知道要从堆里分配多少存储空间和存在时间,所以灵活性比堆栈好。但是,在堆里分配存储空间需花费更长时间
(4) 静态存储区(方法区):在RAM区域,static修饰,程序运行期间,静态存储的数据将随时等候调用。
(5) 常量池:在程序代码内部,final修饰
(6) 非RAM存储:数据完全独立于一个程序之外,程序不运行时仍可存在。比如,硬盘文件
2. 程序举例
以下内容转载自 https://www.cnblogs.com/SaraMoring/p/5687466.html
demo1:
int i1 = 9;
int i2 = 9;
int i3 = 9;
final int INT1 = 9;
final int INT2 = 9;
final int INT3 = 9;
demo2:
String s1 = "china";
String s2 = "china";
String s3 = "china";
String ss1 = new String("china");
String ss2 = new String("china");
String ss3 = new String("china");
解释下3个黄色箭头,对于通过new产生一个字符串(假设为“china”)时,会先去常量池中查找是否已经有了“china”对象,如果没有则在常量池中创建一个此字符串对象,然后堆中再创建一个常量池中此”china”对象的拷贝对象。
面试题:String s=new String("xyz"); 会产生几个对象?答案是一个或两个,如果常量池中原来没有"xyz”,就是两个。demo3:
class BirthDate {
private int day;
private int month;
private int year;
public BirthDate(int d, int m, int y) {
day = d;
month = m;
year = y;
}
// 省略get,set方法………
}
public class Test {
public static void main(String args[]) {
int date = 9;
Test test = new Test();
test.change(date);
BirthDate d1 = new BirthDate(7, 7, 1970);
}
public void change(int i) {
i = 1234;
}
}
对于以上这段代码,date为局部变量,i,d,m,y都是形参为局部变量,day,month,year为成员变量。下面分析一下代码执行时候的变化:
(1) main方法开始执行:
int date = 9; date局部变量,基础类型,引用和值都存在栈中。
(2) Test test = new Test();test为对象引用,存在栈中,对象(new Test())存在堆中。
(3) test.change(date); i为局部变量,引用和值存在栈中。当方法change执行完成后,i就会从栈中消失。
(4) BirthDate d1= new BirthDate(7,7,1970); d1为对象引用,存在栈中,对象(new BirthDate())存在堆中,其中d,m,y为局部变量存储在栈中,且它们的类型为基础类型,因此它们的数据也存储在栈中。day,month,year为成员变量,它们存储在堆中(new BirthDate()里面)。当BirthDate构造方法执行完之后,d,m,y将从栈中消失。
(5) main方法执行完之后,date变量,test,d1引用将从栈中消失,new Test(), new BirthDate()将等待垃圾回收。