文章目录
- 运行时数据区域:
- 内存溢出和内存泄漏:
运行时数据区域:
1、程序计数器:
线程私有。
为了线程切换后,恢复到正确的执行位置。
不会出现OutOfMemoryError。
2、虚拟机栈:
线程私有。
生命周期与线程相同。
服务于字节码文件。(.class文件)
会出现两种情况:
栈内存溢出StackOverFlowError:
发生于方法的无限递归调用。
内存溢出OutOfMemoryError:
程序在申请内存后,无法释放已申请的内存。
3、本地方法栈:
服务于本地方法。
线程私有。
会出现两种情况:
栈内存溢出StackOverFlowError:
发生于方法的无限递归调用。
内存溢出OutOfMemoryError:
程序在申请内存后,无法释放已申请的内存。
4、堆:
内存最大。
线程共享。
垃圾收集器管理的内存区域,又称GC堆。
虚拟机启动时创建。
存放对象实例。
分代收集:
新生代、老年代、永久代...
分代收集的好处:
为了更好地回收内存/更快分配内存。
通过参数扩展大小:
-Xmx、-Xms
物理上不连续的内存空间。
5、方法区:
线程共享。
存放:
类型信息、常量、静态变量、编译后的代码缓存等
JDK8以前:
方法区又称为永久代。
但是这样会造成内存溢出。(永久代有-XX:MaxPermSize的上限)
JDK8:
放弃了永久代。改为元空间代替。
JDK6的时候,有放弃永久代,采用本地内存来实现方法区的计划。
JDK7的时候,将字符串常量池、静态变量移至java堆中。将类型信息移至元空间。
6、常量池:
是方法区的一部分。
用于存放编译期生成的各种字面量和符号引用。
常量池无法再申 请到内存时会抛出OutOfMemoryError异常。
具备动态性;运行期间也可能将新的常量放入池中;用得比较多的便是String类的intern()方法。
7、直接内存:
不属于运行时数据区的一部分。
会频繁使用,会造成OutOfMemoryError。
内存溢出和内存泄漏:
内存溢出OutOfMemoryError:
程序在申请内存时,没有足够的内存空间供其使用。
内存泄漏MemoryLeak:
程序在申请内存后,无法释放已申请的内存。
内存泄漏会最终导致内存溢出。