欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java中的引用類型和使用場景詳細(xì)

 更新時(shí)間:2021年10月08日 14:18:28   作者:Grey  
這篇文章介紹的是Java中的引用類型和使用場景,主要內(nèi)容展開Java中的引用類型,有強(qiáng)引用、軟引用 、弱引用、虛引用,需要的朋友可以參考一下

Java中的引用類型有哪幾種?

Java中的引用類型分成 強(qiáng)引用 , 軟引用 , 弱引用 , 虛引用 。

1、強(qiáng)引用

沒有引用指向這個(gè)對象,垃圾回收會回收

package git.snippets.juc;

import java.io.IOException;

public class NormalRef {
    public static void main(String[] args) throws IOException {
        M m = new M();
        m = null;
        System.gc();
        System.in.read();
    }
    static class M {
        M() {}
        @Override
        protected void finalize() throws Throwable {
            System.out.println("finalized");
        }
    }
}

2、軟引用

當(dāng)有一個(gè)對象被一個(gè)軟引用所指向的時(shí)候,只有系統(tǒng)內(nèi)存不夠用的時(shí)候,才會被回收,可以用做緩存(比如緩存大圖片)

示例如下代碼:注:執(zhí)行以下方法的時(shí)候,需要把VM options設(shè)置為 -Xms20M -Xmx20M 。

package git.snippets.juc;

import java.io.IOException;
import java.lang.ref.SoftReference;
import java.util.concurrent.TimeUnit;

/**
 * heap將裝不下,這時(shí)候系統(tǒng)會垃圾回收,先回收一次,如果不夠,會把軟引用干掉
 * 軟引用,適合做緩存
 * 示例需要把Vm options設(shè)置為:-Xms20M -Xmx20M
 */
public class SoftRef {
    public static void main(String[] args) throws IOException {
        SoftReference<byte[]> reference = new SoftReference<>(new byte[1024 * 1024 * 10]);
        System.out.println(reference.get());
        System.gc();
        try {
            TimeUnit.SECONDS.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(reference.get());
        byte[] bytes = new byte[1024 * 1024 * 10];
        System.out.println(reference.get());
        System.in.read();
    }
}

上述代碼在第一次執(zhí)行 System.out.println(reference.get()) 時(shí)候,由于堆的最大最小值都是 20M ,而我們分配的 byte 數(shù)組是 10M ,沒有超過最大堆內(nèi)存,所以執(zhí)行垃圾回收,軟引用不被回收,后續(xù)又調(diào)用了 byte[] bytes = new byte[1024 * 1024 * 10]; 再次分配了 10M 內(nèi)存,此時(shí)堆內(nèi)存已經(jīng)超過設(shè)置的最大值,會進(jìn)行回收,所以最后一步的 System.out.println(reference.get()); 無法 get 到數(shù)據(jù)。

3、弱引用

只要垃圾回收,就會回收。如果有一個(gè)強(qiáng)引用指向弱引用中的這個(gè)對象,如果這個(gè)強(qiáng)引用消失,這個(gè)對象就應(yīng)該被回收。一般用在容器里面。

代碼示例如下:

package git.snippets.juc;

import java.lang.ref.WeakReference;
import java.util.HashMap;
import java.util.concurrent.TimeUnit;

/**
 * 弱引用遭到gc就會回收
 * ThreadLocal應(yīng)用,緩存應(yīng)用,WeakHashMap
 */
public class WeakRef {
    public static void main(String[] args) {
        WeakReference<T> reference = new WeakReference<>(new T());
        System.out.println(reference.get());
        System.gc();
        System.out.println(reference.get());
    }
    static class T {
        T() {}
        @Override
        protected void finalize() {
            System.out.println("finalized");
        }
    }
}

如果執(zhí)行了一次 GC reference.get() 獲取到的值即為空。

4、弱引用的使用場景

弱引用的一個(gè)典型應(yīng)用場景就是 ThreadLocal ,以下是 ThreadLocal 的的簡要介紹

set方法:

public void set(T value) {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null)
            map.set(this, value);
        else
            createMap(t, value);
    }


get方法:

public T get() {
        Thread t = Thread.currentThread();
        ThreadLocalMap map = getMap(t);
        if (map != null) {
            ThreadLocalMap.Entry e = map.getEntry(this);
            if (e != null) {
                @SuppressWarnings("unchecked")
                T result = (T)e.value;
                return result;
            }
        }
        return setInitialValue();
    }


ThreadLocalMap 是當(dāng)前線程的一個(gè)成員變量,所以,其他線程無法讀取當(dāng)前線程設(shè)置的 ThreadLocal 值。

ThreadLocal.ThreadLocalMap threadLocals = null;


ThreadLocal 的主要應(yīng)用場景

場景一:每個(gè)線程需要一個(gè)獨(dú)享的對象:假設(shè)有100個(gè)線程都需要用到 SimpleDateFormat 類來處理日期格式,如果共用一個(gè) SimpleDateFormat ,就會出現(xiàn)線程安全問題,導(dǎo)致數(shù)據(jù)出錯,如果加鎖,就會降低性能,此時(shí)使用 ThreadLocal ,給每個(gè)線程保存一份自己的本地 SimpleDateFormat ,就可以同時(shí)保證線程安全和性能需求。

場景二:每個(gè)線程內(nèi)部保存全局變量,避免傳參麻煩:假設(shè)一個(gè)線程的作用是拿到前端用戶信息,逐層執(zhí)行 Service1 , Service2 , Service3 , Service4 層的業(yè)務(wù)邏輯,其中每個(gè)業(yè)務(wù)層都會用到用戶信息,此時(shí)一個(gè)解決辦法就是將 User 信息對象作為參數(shù)層層傳遞,但是這樣會導(dǎo)致代碼冗余且不利于維護(hù)。此時(shí)可以將 User 信息對象放入當(dāng)前線程的 Threadlocal 中,就變成了全局變量,在每一層業(yè)務(wù)層中,需要使用的時(shí)候直接從 Threadlocal 中獲取即可。

場景三: Spring 的聲明式事務(wù),數(shù)據(jù)庫連接寫在配置文件,多個(gè)方法可以支持一個(gè)完整的事務(wù),保證多個(gè)方法是用的同一個(gè)數(shù)據(jù)庫連接(其實(shí)就是放在 ThreadLocal 里面)

了解了 ThreadLocal 簡要介紹以后,我們可以深入理解一下 ThreadLocal 的一個(gè)內(nèi)部原理,前面提到, ThreadLocal 的 set 方法實(shí)際上是往當(dāng)前線程的一個(gè) threadLocals 表中插入一條記錄,而這個(gè)表中的記錄都存在一個(gè) Entry 對象中,這個(gè)對象有一個(gè)key和一個(gè)value, key 就是當(dāng)前線程的 ThreadLocal 對象。

static class Entry extends WeakReference<ThreadLocal<?>> {
            /** The value associated with this ThreadLocal. */
            Object value;

            Entry(ThreadLocal<?> k, Object v) {
                super(k);
                value = v;
            }
        }

這個(gè) Entry 對象繼承了 WeakReference , 且構(gòu)造函數(shù)調(diào)用了 super(k) , 所以 Entry 中的 key 是通過一個(gè)弱引用指向的 ThreadLocal ,所以,我們在主方法中調(diào)用

ThreadLocal<Object> tl = new ThreadLocal<>();


tl 是通過強(qiáng)引用指向這個(gè) ThreadLocal 對象。

當(dāng)前線程的 threadLocalMap 中的 key 是通過弱引用指向 ThreadLocal 對象,這樣就可以保證,在 tl 指向空以后,這個(gè) ThreadLocal 會被回收,否則,如果 threadLocalMap 中的 key 是強(qiáng)引用指向 ThreadLocal 對象話,這個(gè) ThreadLocal 對象永遠(yuǎn)不會被回收。就會導(dǎo)致內(nèi)存泄漏。

但是,即便 key 用弱引用指向 ThreadLocal 對象, key 值被回收后, Entry 中的 value 值就無法被訪問到了,且 value 是通過強(qiáng)引用關(guān)聯(lián),所以,也會導(dǎo)致內(nèi)存泄漏,所以,每次在 ThreadLocal 中的對象不用了,記得要調(diào)用 remove 方法,把對應(yīng)的 value 也給清掉。

5、虛引用

用于管理堆外內(nèi)存回收

虛引用關(guān)聯(lián)了一個(gè)對象,以及一個(gè)隊(duì)列,只要垃圾回收,虛引用就被回收,一旦虛引用被回收,虛引用會被裝到這個(gè)隊(duì)列,并會收到一個(gè)通知(如果有值入隊(duì)列,會得到一個(gè)通知)所以,如果想知道虛引用何時(shí)被回收,就只需要不斷監(jiān)控這個(gè)隊(duì)列是否有元素加入進(jìn)來了。

虛引用里面關(guān)聯(lián)的對象用get方法是無法獲取的。

import java.lang.ref.PhantomReference;
import java.lang.ref.Reference;
import java.lang.ref.ReferenceQueue;
import java.util.LinkedList;
import java.util.List;

// 配置 -Xms20M -Xmx20M
public class PhantomRef {
    private static final List<Object> LIST = new LinkedList<>();
    private static final ReferenceQueue<P> QUEUE = new ReferenceQueue<>();


    public static void main(String[] args) {
        PhantomReference<P> phantomReference = new PhantomReference<>(new P(), QUEUE);
        new Thread(() -> {
            while (true) {
                LIST.add(new byte[1024 * 1024]);
                try {
                    Thread.sleep(1000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                    Thread.currentThread().interrupt();
                }
                System.out.println(phantomReference.get());
            }
        }).start();

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

        try {
            Thread.sleep(500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

    }

    static class P {
        @Override
        protected void finalize() throws Throwable {
            System.out.println("finalized");
        }
    }
}

6、虛引用的應(yīng)用場景

JDK的 NIO 包中有一個(gè) DirectByteBuffer , 這個(gè) buffer 指向的是堆外內(nèi)存,所以當(dāng)這個(gè) buffer 設(shè)置為空的時(shí)候,Java的垃圾回收無法回收,所以,可以用虛引用來管理這個(gè) buffer ,當(dāng)我們檢測到這個(gè)虛引用被垃圾回收器回收的時(shí)候,可以做出相應(yīng)的處理,去回收堆外內(nèi)存。

到此這篇關(guān)于Java中的引用類型和使用場景詳細(xì)的文章就介紹到這了,更多相關(guān)Java中的引用類型和使用場景內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 解決Mybatis查詢方法selectById()主鍵不一致問題

    解決Mybatis查詢方法selectById()主鍵不一致問題

    這篇文章主要介紹了解決Mybatis查詢方法selectById()主鍵不一致問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-10-10
  • mybatis 字段名自動轉(zhuǎn)小寫的實(shí)現(xiàn)

    mybatis 字段名自動轉(zhuǎn)小寫的實(shí)現(xiàn)

    這篇文章主要介紹了mybatis 字段名自動轉(zhuǎn)小寫的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-03-03
  • Maven訪問倉庫順序代碼實(shí)例解析

    Maven訪問倉庫順序代碼實(shí)例解析

    這篇文章主要介紹了Maven訪問倉庫順序?qū)嵗馕?文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-08-08
  • java注解結(jié)合aspectj AOP進(jìn)行日志打印的操作

    java注解結(jié)合aspectj AOP進(jìn)行日志打印的操作

    這篇文章主要介紹了java注解結(jié)合aspectj AOP進(jìn)行日志打印的操作,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • Java代理模式實(shí)例分析

    Java代理模式實(shí)例分析

    這篇文章主要介紹了Java代理模式,結(jié)合實(shí)例形式對比分析了java代理模式的使用方法與相關(guān)操作技巧,需要的朋友可以參考下
    2019-07-07
  • Java注解@Transactional事務(wù)類內(nèi)調(diào)用不生效問題及解決辦法

    Java注解@Transactional事務(wù)類內(nèi)調(diào)用不生效問題及解決辦法

    這篇文章主要介紹了Java注解@Transactional事務(wù)類內(nèi)調(diào)用不生效問題及解決辦法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-05-05
  • Java基礎(chǔ)之查找文本特定內(nèi)容后進(jìn)行修改

    Java基礎(chǔ)之查找文本特定內(nèi)容后進(jìn)行修改

    這篇文章主要介紹了Java基礎(chǔ)之查找文本特定內(nèi)容后進(jìn)行修改,文中有非常詳細(xì)的代碼示例,對正在學(xué)習(xí)java基礎(chǔ)的小伙伴們有非常好的幫助,需要的朋友可以參考下
    2021-04-04
  • Java實(shí)現(xiàn)五子棋游戲的完整代碼

    Java實(shí)現(xiàn)五子棋游戲的完整代碼

    這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)五子棋游戲的完整代碼,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-10-10
  • java導(dǎo)出Excel(非模板)可導(dǎo)出多個(gè)sheet方式

    java導(dǎo)出Excel(非模板)可導(dǎo)出多個(gè)sheet方式

    Java開發(fā)中,導(dǎo)出Excel是常見需求,有時(shí)需要支持多個(gè)Sheet導(dǎo)出,此技巧介紹非模板方式實(shí)現(xiàn)單標(biāo)題單Sheet以及多Sheet導(dǎo)出,標(biāo)題一致或不一致均可,可換成Map使用,適合個(gè)人開發(fā)者和需要Excel導(dǎo)出功能的場景
    2024-09-09
  • Java8接口中引入default關(guān)鍵字的本質(zhì)原因詳析

    Java8接口中引入default關(guān)鍵字的本質(zhì)原因詳析

    Default方法是在java8中引入的關(guān)鍵字,也可稱為Virtual extension methods—虛擬擴(kuò)展方法,這篇文章主要給大家介紹了關(guān)于Java8接口中引入default關(guān)鍵字的本質(zhì)原因,需要的朋友可以參考下
    2022-01-01

最新評論