JVM 内存溢出、泄漏问题

1 内存溢出、内存泄漏排查

1.1 内存溢出

内存溢出:是指程序在申请内存时,没有足够的内存空间供其使用,出现OutOfMemoryError错误。

内存溢出的原因可能为:

  • 存在死循环和方法的无限递归调用。
  • 大量循环产生新对象。
  • 内存中加载的数据量过于庞大,如一次从数据库取出过多数据。
  • 集合类中有对对象的引用,使用完后未清空,使得JVM不能回收。(内存泄漏)
  • 可以通过Jstack工具和图形化JConsole工具查询到发生异常的具体区域。
  • 启动参数内存值设定的过小。

1.2 内存泄漏

内存泄漏:是指程序在申请内存后,无法释放已申请的内存空间,也就是说分配出去的内存无法回收。

​ 泄漏的原因本质上是长期存活对象引用短期存活对象,导致短期存活对象占用的对象无法回收。

内存泄漏的几种常景:

  • 静态集合类。像HashMap、Vector等集合的使用最容易出现内存泄露。因为这些集合属于静态集合,这些静态变量的生命周期和应用程序一致,他们所引用的所有的对象Object也不能被释放,因为他们也将一直被Vector等引用着。
  • **各种连接,如数据库连接、网络连接和IO连接等。**在对数据库进行操作的过程中,首先需要建立与数据库的连接,当不再使用时,需要调用close方法来释放与数据库的连接。只有连接被关闭后,垃圾回收器才会回收对应的对象。否则,如果在访问数据库的过程中,对Connection、Statement或ResultSet不显性地关闭,将会造成大量的对象无法被回收,从而引起内存泄漏。
  • 变量不合理的作用域。一般而言,一个变量的定义的作用范围大于其使用范围,很有可能会造成内存泄漏。另一方面,如果没有及时地把对象设置为null,很有可能导致内存泄漏的发生。
  • 内部类持有外部类,如果一个外部类的实例对象的方法返回了一个内部类的实例对象,这个内部类对象被长期引用了,即使那个外部类实例对象不再被使用,但由于内部类持有外部类的实例对象,这个外部类对象将不会被垃圾回收,这也会造成内存泄露。
  • 改变哈希值,当一个对象被存储进HashSet集合中以后,就不能修改这个对象中的那些参与计算哈希值的字段了,否则,对象修改后的哈希值与最初存储进HashSet集合中时的哈希值就不同了,在这种情况下,即使在contains方法使用该对象的当前引用作为的参数去HashSet集合中检索对象,也将返回找不到对象的结果,这也会导致无法从HashSet集合中单独删除当前对象,造成内存泄露。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值