java中的引用類型:強軟弱虛

java中的引用類型共4種:強軟弱虛,具體每種類型的特點和應用場景。記錄下。本文是看了馬士兵老師的視頻後記錄整理的。加深印象。

基本概念

1. 強引用

強引用是使用最普遍的引用。如果一個對象具有強引用,那垃圾回收器絕不會回收它。當內存空間不足時,Java虛擬機寧願拋出OutOfMemoryError錯誤,使程序異常終止,也不會靠隨意回收具有強引用的對象來解決內存不足的問題。

顯式地設置M對象爲null,或讓其超出對象的生命週期範圍,則gc認爲該對象不存在引用,這時就可以回收這個對象
示例代碼

聲明一個M類

public class M {

    /**
     * 當這個對象會被回收的時候,finalize會被調用
     *
     * @throws Throwable
     */
    @Override
    protected void finalize() throws Throwable {
        System.out.println("finalize");
    }
}

強引用調用

public class NormalReference {

    public static void main(String[] args) throws IOException {
        M m = new M();
        m = null;
        System.gc();

        // 如果不寫 main方法退出。System.gc()在垃圾回收線程裏;
        // 有可能還沒來得及回收main方法就退出了
        System.in.read();
    }
}

輸出結果

finalize

2. 軟引用

軟引用對象是在jvm內存不夠的時候纔會被回收

代碼示例

public class Soft {

    public static void main(String[] args) {
        System.out.println(Runtime.getRuntime().totalMemory() / 1024 / 1024);
        //10m
        SoftReference<byte[]> m = new SoftReference<>(new byte[1024 * 1024 * 10]);

        System.out.println(m.get());
        System.gc();
        try {
            Thread.sleep(1500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(m.get());

        // 再分配一個數組,heap裝不下去,這時候系統會垃圾回收,
        // 先回收一次,如果不夠,會把軟引用回收
        byte[] b = new byte[1024 * 1024 * 11];

        System.out.println(m.get());
    }
}

設置程序運行參數:-Xmx20M

運行結果

19
[B@1540e19d
[B@1540e19d
null

我們可以看到,這個時候已經被回收了。

應用場景:軟引用時候做緩存

3. 弱引用

弱引用與軟引用的區別在於:只具有弱引用的對象擁有更短暫的生命週期。在垃圾回收器線程掃描它所管轄的內存區域的過程中,一旦發現了只具有弱引用的對象,不管當前內存空間足夠與否,都會回收它的內存

代碼示例

public class Weak {

    public static void main(String[] args) {
        WeakReference<M> m = new WeakReference<>(new M());

        System.out.println(m.get());
        System.gc();
        System.out.println(m.get());
    }
}

執行結果

M@1540e19d
null
finalize

4. 虛引用

如果一個對象僅持有虛引用,那麼它就和沒有任何引用一樣,在任何時候都可能被垃圾回收器回收。
虛引用必須和引用隊列(ReferenceQueue)聯合使用。當垃圾回收器準備回收一個對象時,如果發現它還有虛引用,就會在回收對象的內存之前,把這個虛引用加入到與之關聯的引用隊列中。

示例

public class Phantom {

    private static final List<Object> LIST = new LinkedList<>();
    private static final ReferenceQueue<M> QUEUE = new ReferenceQueue<>();

    public static void main(String[] args) {
        PhantomReference<M> phantomReference = new PhantomReference<>(new M(), QUEUE);

        new Thread(
                () -> {
                    while (true) {
                        LIST.add(new byte[1024 * 1024]);
                        try {
                            Thread.sleep(1000);
                        } catch (InterruptedException e) {
                            Thread.currentThread().interrupt();
                        }
                        System.out.println(phantomReference.get());
                    }

                }

        ).start();

        new Thread(() -> {
            while (true) {
                Reference<? extends M> poll = QUEUE.poll();
                if (poll != null) {
                    System.out.println("-------虛引用對象被jvm回收了------" + poll);
                }
            }
        }).start();
    }
}

運行結果

finalize
null
Exception in thread "Thread-0" java.lang.OutOfMemoryError: Java heap space
	at Phantom.lambda$main$0(Phantom.java:18)
	at Phantom$$Lambda$1/1078694789.run(Unknown Source)
	at java.lang.Thread.run(Thread.java:748)
-------虛引用對象被jvm回收了------java.lang.ref.PhantomReference@688ee48d


應用場景:堆外內存的管理

總結

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章