内存结构
方法区(Method Area):
虚拟机栈(VM Stack):
一个线程一个栈,一个方法一个栈帧。
本地方法(Native Method Stack):
堆(Heap):
程序计算器(Program Counter Register):
堆内存
堆内存包括新生代(new)和老年代(old)
新生代分一个eden和两个survivor(幸存者)
老年代分tenured
垃圾对象
什么是垃圾对象?
如何确定垃圾对象?
引用计数:存在有循环引用的问题。
正向可达:从roots对象计算可以到达的对象。
垃圾收集算法
Mark-Sweep标记清除
问题:碎片化
Copying复制
问题:浪费内存
新生代应该比较多。survivor两个之间相互copy
Mark-Compact标记压缩
问题:效率比copy略低
老年代应用比较多。
新生代(new):存活对象少,使用copying算法,占用内存空间也不大,效率高。
老年代(old):垃圾少,一般使用mark-copact。
垃圾收集器
- Serial Collector(串行垃圾回收器)
- 单线程
- -XX:+UseSerialGC
- Parallel Collector(并行垃圾回收器)
- 多线程,并发量大,不过每次垃圾收集,jvm要暂停
- -XX:UseParallelGC
- CMS Collector(并发垃圾回收器)
- XX:+USeParNewGC
- 多线程,并发量大,GC线程和应用线程都在运行
- G1(垃圾回收器)
- 不仅停顿短,同时并发量大
- –XX:+UseG1GC
jvm参数
1、-:标准参数,所有jvm都应该支持
2、-X:非标准,每个jvm实现都不同
3、-XX:不稳定参数,下一个版本可能会取消
java对象的分配
- 栈上分配
- 线程私有小对象
- 无逃逸
- 支持标量替换
- 无需调整
- 线程本地分配TLAB(Thread Local Allcation Buffer)
- 占用eden、默认1%
- 多线程的时候不用竞争eden就可以申请空间,提高效率
- 小对象
- 无需调整
- 老年代
- 大对象
- eden
jvm调优
- 线程大小(栈内存大小)
设置:Xss:128
栈内存越大,栈帧数量就越多,方法调用就越深,但线程并发量就少
栈内存越小,栈帧数量就越少,方法调用就越浅,但线程并发量就多
持久加压,测峰值
jmx
使用工具
visualvm:java内存监控;
jvisualvm.exe:java自动工具,bin目录下。
apache-jmeter:请求模拟工具。