Java内存区域与内存溢出异常
java与c++之间有一堵内存动态分配和垃圾收集技术所围成的高墙
1.1.1 程序计数器
- 程序计数器是一块较小的内存空间,可以看做当前线程所执行的字节码行号指示器。
- java虚拟机的多线程是通过线程轮流切换、分配处理器执行时间的方式来实现的。一个处理器只会执行一条线程中的指令。每条线程都有一个独立的程序计数器,各条线程之间技术器互不影响,独立存储。
1.1.2 java虚拟机栈
- java虚拟栈是线程私有的,生命周期与线程相同。虚拟机栈描述的是java方法执行的线程内存模型;每个方法被执行的时候,java虚拟机栈都会同步创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法别调用至执行完毕的过程,就对着一个栈帧在虚拟机栈从入栈到出栈的过程。
- 局部变量表存放了编译器可知的各种java基本数据类型、对象引用(reference类型)和returnAdress类型
1.1.3 本地方法栈
与虚拟机栈发挥的作用非常相似。
1.1.4 java堆
- java堆是虚拟机所管理的内存最大的一块。是被所有线程共享的一块内存区域
- 无论从什么角度,无论如何划分,都不会改变java堆中存储内容的共性,无论是那个区域,存储的只能是对象的实例
1.1.5 方法区
- 方法区与java堆一样,是各个线程共享的区域,用于存储已被虚拟机加载的类型信息、常量、静态变量、即时编译器编译后的代码缓存等数据
1.1.6运行时常量池
- 是方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息常量表,用于存储编译期产生的生成的各种字面量与符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中。
2.1.1 对象的内存布局
-
对象在内存中的存储布局可以划分为三个部分:对象头,实例数据和对齐填充
-
对象头包括两类信息:第一类是用于存储对象自身的运行时数据。如哈希码、GC分代年龄、锁状态标志、线程持有的锁、偏向线程ID、偏向时间戳等。另一部分是类型指针
异常
如果线程请求的深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;如果java虚拟机容量可以动态扩展,当栈扩展时无法申请到足够的内存会抛出OutOfMemoryError异常。
OutOfMemoryError异常
- 除了程序计数器外,虚拟机内存的其他几个运行时区域都有发生OutOfMemoryError异常的可能
- 出现java堆内存溢出时,异常堆栈信息"java.lang.OutOfMemoryError会跟随进一步提示"java heap space"
- 常规的处理方法是首先通过内存映像分析工具对Dump出来的堆转储快照进行分析。第一步首先确认内存中导致oom的对象是否是必要的,也就是要先分析出到底是出现了内存泄露还是内存溢出。