自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

發(fā)現(xiàn)Java虛擬機(jī)內(nèi)存泄露問題

原創(chuàng)
開發(fā) 后端
內(nèi)存泄露問題對(duì)于程序來說,是一個(gè)很難發(fā)現(xiàn),并且容易引起嚴(yán)重災(zāi)害的事情。Java一直以其垃圾回收機(jī)制為自豪,那是否這種機(jī)制就是完美的呢。

常規(guī)理解上,Java的內(nèi)存管理機(jī)制是將局部變量保存在堆中,當(dāng)變量的作用域結(jié)束之后,該變量所占用的內(nèi)容會(huì)被自動(dòng)回收。不需要做任何特殊的處理。比如下面的代碼:

  1. public class JavaMemory{  
  2.     private final int dataSize = (int) (Runtime.getRuntime().maxMemory() * 0.6);  
  3.     public void f(){  
  4.         {  
  5.             byte[] data1 = new byte[dataSize];  
  6.         }  
  7.         byte[] data2 = new byte[dataSize];  
  8.     }  
  9.  
  10.     public static void main(String[] args) {  
  11.         JavaMemoryPuzzle jmp = new JavaMemoryPuzzle();  
  12.         jmp.f();  
  13.     }  

在這個(gè)例子中,方法f()里定義了兩個(gè)局部變量,變量data1和data2的作用域不同。按照正常理解,雖然兩各個(gè)數(shù)組所需要的內(nèi)存之和已經(jīng)超過了可用內(nèi)存,但是因?yàn)閐ata1會(huì)被及時(shí)回收,不會(huì)出現(xiàn)內(nèi)存溢出錯(cuò)誤。

如果我們實(shí)際執(zhí)行這個(gè)例子,會(huì)發(fā)現(xiàn)出現(xiàn)了java.lang.OutOfMemoryError錯(cuò)誤。這是為什么?如果在BEA或者IBM的虛擬機(jī)上測(cè)試過這個(gè)例子,并不會(huì)出現(xiàn)錯(cuò)誤。也就是說,SUN的JVM在內(nèi)存回收機(jī)制上存在漏洞或者BUG。

這個(gè)問題該如何修正呢,方法其實(shí)很簡單。只需要在變量作用域結(jié)束之前,將變量置為空就可以了。修改之后的結(jié)果如下:

  1. public class JavaMemory{  
  2.     private final int dataSize = (int) (Runtime.getRuntime().maxMemory() * 0.6);  
  3.     public void f(){  
  4.         {  
  5.             byte[] data1 = new byte[dataSize];  
  6.             data1 = null;  
  7.         }  
  8.         byte[] data2 = new byte[dataSize];  
  9.     }  
  10.  
  11.     public static void main(String[] args) {  
  12.         JavaMemoryPuzzle jmp = new JavaMemoryPuzzle();  
  13.         jmp.f();  
  14.     }  

發(fā)現(xiàn)這個(gè)問題,對(duì)于Java開發(fā)者來說也許會(huì)很緊張,擔(dān)心自己的代碼是否會(huì)出現(xiàn)同樣問題。大家盡可放心,連續(xù)出現(xiàn)兩個(gè)變量占用內(nèi)存之和超過內(nèi)存限制的情況概率非常小。并且在兩個(gè)變量之間,如果定義了其他變量也不會(huì)出現(xiàn)這個(gè)問題。如下面的代碼就不會(huì)出現(xiàn)問題:

  1. public class JavaMemory{  
  2.     private final int dataSize = (int) (Runtime.getRuntime().maxMemory() * 0.6);  
  3.     public void f(){  
  4.         {  
  5.             byte[] data1 = new byte[dataSize];  
  6.         }  
  7.     int i=1;  
  8.         byte[] data2 = new byte[dataSize];  
  9.     }  
  10.  
  11.     public static void main(String[] args) {  
  12.         JavaMemoryPuzzle jmp = new JavaMemoryPuzzle();  
  13.         jmp.f();  
  14.     }  

 

【編輯推薦】

  1. 專題:Java線程
  2. Java Swing多線程死鎖問題解析
  3. 有趣的Java對(duì)象序列化緩存問題
  4. Java實(shí)用技巧:當(dāng)不能拋出checked異常時(shí)
  5. Java編程語言的優(yōu)點(diǎn)快遞
責(zé)任編輯:佚名 來源: 51CTO
相關(guān)推薦

2019-11-20 15:02:45

Java虛擬機(jī)內(nèi)存

2020-03-10 09:30:11

Java內(nèi)存數(shù)據(jù)

2009-10-09 09:42:07

虛擬機(jī)內(nèi)存

2019-04-09 15:56:46

虛擬機(jī)內(nèi)存區(qū)Java

2011-05-26 15:41:25

java虛擬機(jī)

2009-06-18 13:51:07

Java虛擬機(jī)

2010-09-17 15:47:14

TOMCATJVM

2013-12-23 09:25:21

2018-03-28 14:58:42

虛擬機(jī)內(nèi)核系統(tǒng)

2011-12-28 13:38:00

JavaJVM

2012-08-14 10:05:08

2009-06-17 15:39:00

本機(jī)內(nèi)存硬件限制虛擬內(nèi)存

2024-10-31 09:24:42

2017-02-21 16:40:16

Android垃圾回收內(nèi)存泄露

2009-06-04 16:27:39

Java虛擬機(jī)JVMGC

2020-06-23 09:48:09

Python開發(fā)內(nèi)存

2010-09-25 11:32:24

Java內(nèi)存泄漏

2021-04-14 18:58:01

虛擬機(jī) Java內(nèi)存

2009-06-16 11:11:07

Java內(nèi)存管理Java內(nèi)存泄漏

2009-06-17 15:46:36

Java運(yùn)行時(shí)本機(jī)內(nèi)存
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)