JVM以hotspot 虚拟机为例,由方法区,堆,虚拟机栈,程序计数器,本地方法区,如下图所示
1. Program Counter Register 程序计数器(寄存器) 作用,是记住下一条jvm指令的执行地址,因为CPU在执行程序的时候会在多个线程之间进行切换,所以需要保存当前线程的执行地址
特点: a.是线程私有的 (每个线程都应该有自己的程序计数器,程序计数器是线程私有的。)
b.不会存在内存溢出
0: getstatic #20 // PrintStream out = System.out;
3: astore_1 // --
4: aload_1 // out.println(1);
5: iconst_1 // --
6: invokevirtual #26 // --
9: aload_1 // out.println(2);
10: iconst_2 // --
11: invokevirtual #26 // --
14: aload_1 // out.println(3);
15: iconst_3 // --
16: invokevirtual #26 // --
19: aload_1 // out.println(4);
20: iconst_4 // --
21: invokevirtual #26 // --
24: aload_1 // out.println(5);
25: iconst_5 // --
26: invokevirtual #26 // --
29: return
2. 虚拟机栈
虚拟机栈是给线程使用的,每个线程对应一个虚拟机栈,每个虚拟机栈里面又装了很多的栈帧,栈帧是线程调用某个方法时所分配的内存,每个方法对应一个栈帧,每个线程只能有一个活动栈帧,对应着当前正在执行的那个方法
1. 垃圾回收是否涉及栈内存?
2. 栈内存分配越大越好吗?
3. 方法内的局部变量是否线程安全?
如果方法内局部变量没有逃离方法的作用访问,它是线程安全的 如果是局部变量引用了对象,并逃离方法的作用范围,需要考虑线程安全
栈内存溢出(stack over flow 异常)
1.栈帧过多导致栈内存溢出
2.栈帧过大导致栈内存溢出
3. 本地方法栈
是调用底层的一些方法所需要的内存
4.堆
通过 new 关键字,创建对象都会使用堆内存
特点:
它是线程共享的,堆中对象都需要考虑线程安全的问题
有垃圾回收机制
堆内存溢出是(out of memory): 发生在对象之间相互引用