一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務器之家:專注于服務器技術及軟件下載分享
分類導航

PHP教程|ASP.NET教程|JAVA教程|ASP教程|

服務器之家 - 編程語言 - JAVA教程 - 深入理解Java垃圾回收機制以及內存泄漏

深入理解Java垃圾回收機制以及內存泄漏

2020-05-08 11:32jingxian JAVA教程

下面小編就為大家帶來一篇深入理解Java垃圾回收機制以及內存泄漏。小編覺得挺不錯的,現在就分享給大家,也給的大家做個參考。一起跟隨小編過來看看吧

前言

在segmentfault上看到一個問題:java有完善的GC機制,那么在java中是否會出現內存泄漏的問題,以及能否給出一個內存泄漏的案例。本問題視圖給出此問題的完整答案。

垃圾回收機制簡介

在程序運行過程中,每創建一個對象都會被分配一定的內存用以存儲對象數據。如果只是不停的分配內存,那么程序遲早面臨內存不足的問題。所以在任何語言中,都會有一個內存回收機制來釋放過期對象的內存,以保證內存能夠被重復利用。

內存回收機制按照實現角色的不同可以分為兩種,一種是程序員手動實現內存的釋放(比如C語言)另一種則是語言內建的內存回收機制比如本文將要介紹的java垃圾回收機制。

Java的垃圾回收機制

在程序的運行時環境中,java虛擬機提供了一個系統級的垃圾回收(GC,Carbage Collection)線程,它負責回收失去引用的對象占用的內存。理解GC的前提是理解一些和垃圾回收相關的概念,下文一一介紹這些概念。

對象在jvm堆區的狀態

Java對象的實例存儲在jvm的堆區,對于GC線程來說,這些對象有三種狀態。

1. 可觸及狀態:程序中還有變量引用,那么此對象為可觸及狀態。

2. 可復活狀態:當程序中已經沒有變量引用這個對象,那么此對象由可觸及狀態轉為可復活狀態。CG線程將在一定的時間準備調用此對象的finalize方法(finalize方法繼承或重寫子Object),finalize方法內的代碼有可能將對象轉為可觸及狀態,否則對象轉化為不可觸及狀態。

3. 不可觸及狀態:只有當對象處于不可觸及狀態時,GC線程才能回收此對象的內存。

深入理解Java垃圾回收機制以及內存泄漏

 

GC為了能夠正確釋放對象,必須監控每一個對象的運行狀態,包括對象的申請、引用、被引用、賦值等,GC都需要進行監控,所以無論一個對象處于上文中的任何狀態GC都會知道。

上文說到,GC線程會在一定的時間執行可復活狀態對象的finalize方法,那么何時執行呢?由于不同的JVM實現者可能使用不同的算法管理GC,所以在任何時候,開發者無法預料GC線程進行各項操作(包括檢測對象狀態、釋放對象內存、調用對象的finalize方法)的時機。雖然可以通過System.gc()和Runtime.gc()函數提醒GC線程盡快進行垃圾回收操作,但是這也無法保證GC線程馬上就會進行相應的回收操作。

內存泄露

內存泄漏指由于錯誤的設計造成程序未能釋放已經不再使用的內存,造成資源浪費。GC會自動清理失去引用的對象所占用的內存。但是,由于程序設計錯誤而導致某些對象始終被引用,那么將會出現內存泄漏。

比如下面的例子。使用數組實現了一個棧,有入棧和出棧兩個操作。

 

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
import com.sun.javafx.collections.ElementObservableListDecorator;
import com.sun.swing.internal.plaf.metal.resources.metal_sv;
 
import java.beans.ExceptionListener;
import java.util.EmptyStackException;
 
/**
 * Created by peng on 14-9-21.
 */
public class MyStack {
  private Object[] elements;
  private int Increment = 10;
  private int size = 0;
 
  public MyStack(int size) {
    elements = new Object[size];
  }
 
  //入棧
  public void push(Object o) {
    capacity();
    elements[size++] = o;
  }
 
  //出棧
  public Object pop() {
    if (size == 0)
      throw new EmptyStackException();
    return elements[--size];
  }
 
  //增加棧的容量
  private void capacity() {
    if (elements.length != size)
      return;
    Object[] newArray = new Object[elements.length + Increment];
    System.arraycopy(elements, 0, newArray, 0, size);
  }
 
  public static void main(String[] args) {
    MyStack stack = new MyStack(100);
    for (int i = 0; i < 100; i++)
      stack.push(new Integer(i));
    for (int i = 0; i < 100; i++) {
      System.out.println(stack.pop().toString());
    }
  }
}

這個程序是可用的,支持常用的入棧和出棧操作。但是,有一個問題沒有處理好,就是當出棧操作的時候,并沒有釋放數組中出棧元素的引用,這導致程序將一直保持對這個Object的引用(此object由數組引用),GC永遠認為此對象是可觸及的,也就更加談不上釋放其內存了。這就是內存泄漏的一個典型案例。針對此,修改后的代碼為:

?
1
2
3
4
5
6
7
8
//出棧
  public Object pop() {
    if (size == 0)
      throw new EmptyStackException();
    Object o = elements[--size];
    elements[size] = null;
    return o;
  }

以上這篇深入理解Java垃圾回收機制以及內存泄漏就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支持服務器之家。

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 成人免费草草视频 | 国产精亚洲视频 | 1769在线视频 | 天天狠天天透天干天天怕处 | 午夜神器老司机高清无码 | 黑人好大好硬好深好爽想要h | 亚洲天堂中文 | 91视频综合网 | 四虎精品影视 | 青青青久在线视频免费观看 | 亚洲国产一区二区三区青草影视 | 午夜五月天| 成人黄色a级片 | 国产精品亚洲精品青青青 | 欧美日韩国产中文字幕 | 久久精品男人影院 | 范冰冰性xxxxhd | 俄罗斯三级在线观看级 | 经典千人斩一区二区视频 | 欧美 亚洲 一区 | 色中文| 日本一区二区免费在线观看 | 亚洲四虎影院 | 黑人群性xxx | 34看网片午夜理 | 拔插拔插.com | 欧美精品黑人巨大在线播放 | 日韩伦理一区 | 91免费破解版 | 美女国内精品自产拍在线播放 | 国产精品不卡 | 全黄h全肉细节文在线观看 全彩成人18h漫画 | 波多野结衣xxxxx在线播放 | 精品一区heyzo在线播放 | 99热.com | 国产精品久久久久久五月尺 | 日本视频观看 | 欧美成人aaaa免费高清 | 18hdxxxx中国 | 国内自拍网红在线综合 | 成人精品网 |