1、JVM的内存空间分为:程序计数器,方法区,虚拟机栈,本地方法栈,java堆。
2、程序计数器很小,用来记录下一条要执行的指令
3、java虚拟机规范允许java栈的大小是动态的或者是固定的,如果线程执行过程中,请求的栈深度大于最大可用的栈深度,则抛出StackOverFlow;如果可以支持动态扩展而在扩展过程中内存不足,则Outofmemory异常。可以通过参数-Xss来设定栈的大小
4、方法区中重要的是类型信息,常量池,域信息,方法信息。方法区也称为永久区。
回收类元数据的基本条件至少有:所有该类的实例被回收,且装载该类的Classloader被回收
对常量池的回收则很明确,没有被任何引用则可以回收
5、java的堆中存放运行过程中生成的对象等信息
垃圾回收的算法:
1、引用计数法 无法处理循环引用的问题
2、标记-清除算法 可能产生的最大问题是空间碎片
3、复制算法 缺点是将系统内存折半
4、标记-压缩算法 先做标记,然后将存活对象移动到内存的一端,之后清理其他所有的空间
垃圾回收:
一般现在的HotSpot虚拟机广泛使用分代的思想,根据不同的数据特点将内存分为新生代和老年代。
新生代又分为oden,from,to三个区,新生代因为存活对象可能较少,因此可采用复制算法
老年代因为大部分对象可能还在存活,因此可采用标记-压缩算法
老年代的full gc 要比新生代的gc的代价要高得多
一般新的对象会被分配到新生代,但是大对象可能直接进入老年代
优化措施:
1、适当的调整新生代和老年代的大小,如对于大量数据经常需要更新的,则可以加大新生代的大小,尽量减少老年代的full gc;而对于有较多需要长期保持不变的对象,可以适当增大老年代的大小
2、减少频繁使用大对象,因为大对象会直接分配到老年代,如果频繁的生成及回收大对象就会导致老年代不断的gc,这样代价很高。