7 分析java堆
常见内存溢出(OOM)
堆溢出
直接内存溢出
java NIO支持直接内存的使用,也就是通过一段java代码,获得一块堆外的内存空间,这块空间是直接向操作系统申请的。
直接内存不一定能触发GC,除非直接内存使用量达到了-XX:MaxDirectMemorySize的设置,所以保证直接内存不溢出的方法是设定一个系统实际可达的-XX:MaxDirectMemorySize值(默认情况下等于-Xmx的设置)。
过多线程导致OOM
每一个线程的开启都要占用系统内存,当线程太多,也有可能导致OOM。
解决方法一种是减少堆空间,操作系统就可以预留更多内存用于创建线程。另一种是-Xss参数可以指定线程的栈空间。减少栈空间理论上可以容纳更多的线程,但是栈溢出的风险会相应的上升。
永久区溢出
类的数量太多。
GC效率太低引起OOM
不常见,因为在绝大多数场合,效率过低的GC会导致堆溢出。
String在虚拟机中的实现
String对象的三个基本特点
- 不变性
- 针对常量池优化
- 类的final定义
不变性指String对象一旦生成,则不能对它进行改变。
当两个String对象拥有相同的值时,它们只引用常量池中的同一个拷贝。当一个字符串反复出现时,这个技术可以大幅度节省内存空间。
String str1 = new String("abc");
String str2 = new String("abc");
str1==str2 //false
str1==str2.intern(); //false
"abc"==str3.intern(); //true
String.intern()返回字符串在常量池中的引用。
有关String常量池的位置
在jdk1.6之前,这块区域属于永久区的一部分,但在jdk1.7之后,它就被移到了堆中进行管理。