消除過期的對象引用

你能看出以下代碼哪里內(nèi)存泄漏嗎堪嫂?

// Can you spot the "memory leak"?
public class Stack {
  private Object[] elements;
  private int size = 0;
  private static final int DEFAULT_INITIAL_CAPACITY = 16;

  public Stack() {
       elements = new Object[DEFAULT_INITIAL_CAPACITY];
  }

  public void push(Object e) {
    ensureCapacity();
    elements[size++] = e;
  }

  public Object pop() {
    if (size == 0)
         throw new EmptyStackException();
    return elements[--size];
  }

/**
* Ensure space for at least one more element, roughly
* doubling the capacity each time the array needs to grow.
*/
  private void ensureCapacity() {
    if (elements.length == size)
         elements = Arrays.copyOf(elements, 2 * size + 1);
    }
}

答案是:

pop()方法存在內(nèi)存泄漏藻肄。

內(nèi)存泄漏可以稱為“ 無意識的對象保持(unintentional object retention)”。
在pop()方法中從棧中彈出來的對象將不會被當(dāng)做垃圾回收间学。棧內(nèi)部維護著對這些對象的過期引用(obsolete reference)殷费。所謂的過期引用,是指永遠(yuǎn)也不會再被解除的引用低葫。凡是在elements數(shù)組的“活動部分”(active portion)之外的任何引用都是過期的详羡。活動部分是指elements中下標(biāo)小于size的那些元素氮采。

解決方法:上述述例子中的Stack類而言殷绍,只要一個單元被彈出棧,指向它的引用就過期了鹊漠。一旦數(shù)組元素變成了非活動部分的一部分主到,就手工清空這些數(shù)組元素。修改后的pop()方法如下:

public Object pop() {
    if (size == 0)
        throw new EmptyStackException();
    Object result = elements[--size];//result相當(dāng)于一個temp躯概。
    elements[size] = null; // Eliminate obsolete reference
    return result;
}

清空對象引用應(yīng)該是一種例外登钥,而不是一種規(guī)范行為。消除過期引用最好的方法是讓包含該引用的變量結(jié)束其生命周期娶靡。

延伸閱讀

length牧牢、size()、length()的區(qū)別:

  • 數(shù)組的長度(length):數(shù)組能容納元素個數(shù)的值
  • 泛型集合的大凶硕А(size()):泛型中元素的個數(shù)
  • 字符串的長度(length()):字符的個數(shù)

前綴遞減和后綴遞增:

  • 前綴遞減塔鳍,"--"操作符位于變量或表達式前,先執(zhí)行運算呻此,再生成值轮纫。如上例中elements[--size],size大小先減1,所以Object result = elements[--size];中result元素下標(biāo)為size=size-1焚鲜。
  • 后綴遞增掌唾,"++"操作符位于變量或表達式后,先生成值忿磅,再執(zhí)行運算糯彬。如上例中elements[size++] = e;,元素下標(biāo)為size葱她,再執(zhí)行運算size=size+1撩扒。

Arrays.copyOf()方法:

  • a copy of the original array, truncated or padded with nulls to obtain the specified length(原始數(shù)組的副本,縮短或填補null來獲取指定的長度)。
  • 作用:如果數(shù)組元素的個數(shù)等于數(shù)組的長度吨些,新建副本數(shù)組搓谆,將長度擴大為兩倍加一,將數(shù)組副本賦值給elements锤灿。

參考資料

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末挽拔,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子但校,更是在濱河造成了極大的恐慌螃诅,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,640評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件状囱,死亡現(xiàn)場離奇詭異术裸,居然都是意外死亡,警方通過查閱死者的電腦和手機亭枷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,254評論 3 395
  • 文/潘曉璐 我一進店門袭艺,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人叨粘,你說我怎么就攤上這事猾编×龆茫” “怎么了?”我有些...
    開封第一講書人閱讀 165,011評論 0 355
  • 文/不壞的土叔 我叫張陵答倡,是天一觀的道長轰传。 經(jīng)常有香客問我,道長瘪撇,這世上最難降的妖魔是什么获茬? 我笑而不...
    開封第一講書人閱讀 58,755評論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮倔既,結(jié)果婚禮上恕曲,老公的妹妹穿的比我還像新娘。我一直安慰自己渤涌,他們只是感情好佩谣,可當(dāng)我...
    茶點故事閱讀 67,774評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著歼捏,像睡著了一般稿存。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上瞳秽,一...
    開封第一講書人閱讀 51,610評論 1 305
  • 那天瓣履,我揣著相機與錄音,去河邊找鬼练俐。 笑死袖迎,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的腺晾。 我是一名探鬼主播燕锥,決...
    沈念sama閱讀 40,352評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼悯蝉!你這毒婦竟也來了归形?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,257評論 0 276
  • 序言:老撾萬榮一對情侶失蹤鼻由,失蹤者是張志新(化名)和其女友劉穎暇榴,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蕉世,經(jīng)...
    沈念sama閱讀 45,717評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡蔼紧,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,894評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了狠轻。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片奸例。...
    茶點故事閱讀 40,021評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖向楼,靈堂內(nèi)的尸體忽然破棺而出查吊,到底是詐尸還是另有隱情谐区,我是刑警寧澤,帶...
    沈念sama閱讀 35,735評論 5 346
  • 正文 年R本政府宣布菩貌,位于F島的核電站卢佣,受9級特大地震影響重荠,放射性物質(zhì)發(fā)生泄漏箭阶。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,354評論 3 330
  • 文/蒙蒙 一戈鲁、第九天 我趴在偏房一處隱蔽的房頂上張望仇参。 院中可真熱鬧,春花似錦婆殿、人聲如沸诈乒。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,936評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽怕磨。三九已至,卻和暖如春消约,著一層夾襖步出監(jiān)牢的瞬間肠鲫,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,054評論 1 270
  • 我被黑心中介騙來泰國打工或粮, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留导饲,地道東北人。 一個月前我還...
    沈念sama閱讀 48,224評論 3 371
  • 正文 我出身青樓氯材,卻偏偏與公主長得像渣锦,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子氢哮,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,974評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 如圖所示的例子袋毙,這段程序中沒有明顯的錯誤,但是存在一個隱藏的問題(“內(nèi)存泄漏”)冗尤,隨著垃圾回收活動的增加听盖,或者由于...
    郭_4d5f閱讀 436評論 0 0
  • 問題的引出 這段程序有一個“內(nèi)存泄露”,隨著GC活動的增加生闲,或者由于內(nèi)存占用的不斷增加媳溺,程序性能降低會逐漸表現(xiàn)出來...
    每天學(xué)點編程閱讀 518評論 0 4
  • 什么是過期的對象引用? 我們通過簡單的棧實現(xiàn)來引入過期的對象引用碍讯。 實際上悬蔽,這段程序中并沒有很明顯的錯誤。無論如何...
    大海孤了島閱讀 946評論 1 2
  • 很多人可能在想這么一個問題:Java有垃圾回收機制捉兴,那么還存在內(nèi)存泄露嗎蝎困?答案是肯定的录语,所謂的垃圾回收GC會自動管...
    Ruheng閱讀 1,661評論 0 5
  • 第6條:消除過期的對象引用 1. 為什么要消除過期的對象引用 java雖然有自己的垃圾回收機制,但是并沒有那么的智...
    想飛的僵尸閱讀 696評論 0 2