7 分析java堆

来源:互联网 发布:500px哪些摄影师 知乎 编辑:程序博客网 时间:2024/06/08 19:23

7 分析java堆

常见内存溢出(OOM)

堆溢出

直接内存溢出

java NIO支持直接内存的使用,也就是通过一段java代码,获得一块堆外的内存空间,这块空间是直接向操作系统申请的。

直接内存不一定能触发GC,除非直接内存使用量达到了-XX:MaxDirectMemorySize的设置,所以保证直接内存不溢出的方法是设定一个系统实际可达的-XX:MaxDirectMemorySize值(默认情况下等于-Xmx的设置)。

过多线程导致OOM

每一个线程的开启都要占用系统内存,当线程太多,也有可能导致OOM。

解决方法一种是减少堆空间,操作系统就可以预留更多内存用于创建线程。另一种是-Xss参数可以指定线程的栈空间。减少栈空间理论上可以容纳更多的线程,但是栈溢出的风险会相应的上升。

永久区溢出

类的数量太多。

GC效率太低引起OOM

不常见,因为在绝大多数场合,效率过低的GC会导致堆溢出。

String在虚拟机中的实现

String对象的三个基本特点

  1. 不变性
  2. 针对常量池优化
  3. 类的final定义

不变性指String对象一旦生成,则不能对它进行改变。

当两个String对象拥有相同的值时,它们只引用常量池中的同一个拷贝。当一个字符串反复出现时,这个技术可以大幅度节省内存空间。

String str1 = new String("abc");String str2 = new String("abc");str1==str2 //falsestr1==str2.intern(); //false"abc"==str3.intern(); //true

String.intern()返回字符串在常量池中的引用。

有关String常量池的位置

在jdk1.6之前,这块区域属于永久区的一部分,但在jdk1.7之后,它就被移到了堆中进行管理。

原创粉丝点击