链接:docs.oracle.com/javase/8/docs/technotes/tools/unix/java.html
1.一个JVM实例只存在一个堆空间,堆也是Java内存管理的核心区域。
2.Java堆区在Java启动的时候即被创建,其空间大小也就确定了。是JVM管理的最大的一块内存空间。
堆内存的大小是可以调节!
3.堆可以处于物理上不连续的内存空间中,但在逻辑上它应该被视为连续的。
4,所有的线程共享Java堆,在这里还可以划分线程私有的缓冲区(TLAB)!这里是线程安全的!包含在Eden区,仅占整个Eden空间的1%;
开发人员可以通过选项”-XX:+UseTLAB“设置是否开启TLAB空间 默认开启
-XX:TLABWasteTargetPercent设置TLAB空间所占用Eden空间的百分比大小
5.所有的对象实例以及数组都应当在运行时分配在堆上!
6.数组和对象可能永远不会存储在栈上,因为栈帧中保存引用,这个引用指向对象或者数组在堆中的位置!
7.在方法结束后,在堆中的对象不会马上被移除,仅仅在垃圾收集的时候才会被移除。
8.堆,是GC执行垃圾回收的重点区域!
9.现代垃圾收集器大部分都基于分代收集理论,对空间细分为:
新生区(eden+from+to)+老年区+元空间(永久代)
-XX:+PrintGCDetails
堆空间大小的设置
堆空间大小:-Xms10m(初始化内存 只包括年轻代+老年代) -Xmx10m(最大内存)
-X 是jvm的运行参数
ms 是memory start
默认推空间大小:
初始内存大小:物理电脑内存大小的1/64
最大内存大小:物理电脑内存大小1/4
返回Java虚拟机中的堆内存总量:
long initialMemory = Runtime.getRuntime().totalMemory()/1024/1024
返回Java虚拟机试图使用的最大内存:
long maxMemory = Runtime.getRuntime().maxMemory()/1024/1024
3,建议初始堆内存和最大堆内存设置相同的值!
频繁扩容,避免频繁GC对系统造成额外的压力
查看设置的参数:
方式一:jps->jstat -gc 进程id
方式二:-XX:+PrintGCDetails
存储在JVM,中的Java对象可以被划分为两类:
一类是生命周期较短瞬时对象,这类对象的创建和消亡都非常迅速
另外一类对象的生命周期却非常长,在某些极端的情况下还能够与JVM的生命周期保持一致。
Java堆区进一步细分的话:可以划分为年轻代和老年代!
其中年轻代又可以划分为Eden空间、Survivor0空间和Survivor空间(也成为from区,to区)
配置新生代与老年代结构占比:
默认-XX:NewRatio=2,表示新生代占1,老年代占2,新生代占整个堆的1/3
可以修改为-XX:NewRatio=4,表示新生代占1,老年代占4,新生代占整个堆的1/5
在HotSpot中,Eden空间和另外两个Survivor空间缺省所占比例是8:1:1
关闭自适应机制:-XX:+UseAdaptiveSizePolicy
当然开发人员可以通过选项“-XX:SurvivorRatio”调整这个空间比例。比如:-XX:SurvivorRatio=8
几乎所有的Java对象都是在Eden区被new出来的。
如果对象内存太大会直接进入老年区
绝大部分的Java对象的销毁都在新生代进行了。
可以使用选项-Xmn设置新生代最大内存大小
这个参数正常使用默认值即可
常用设置:
-XX:+PrintFlagsInitial:查看所有的参数的默认初始值;
-XX:+PrintFlagsFinal:查看所有的参数的最终数;
-Xmn:设置新生代大小
-XX:DoEscapeAnalysis:启用逃逸分析
-XX:EliminateAllocations:开启标量替换,允许将对象打散分配在栈上
堆是分配对象存储的位置选择嘛?
在Java虚拟机中,对象是在Java堆中分配内存的,但有一种特殊情况,那就是如果经过逃逸分析后发现,一个对象并没有逃逸出去的话,那么就可能被优化成栈上分配。
这是一种可以有效减少Java程序中同步负载和内存堆分配压力的阔函数全局数据流分析算法;
逃逸分析的基本行为就是对象动态作用域:
当一个对象在方法中被定义后,对象只在方法内部使用,则认为没有发生逃逸。则使用栈上分配;
当一个对象在方法被定义后,它被外部方法所引用,则认为发生逃逸。例如作为调用参数传递到其他地方中。