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

Java怎樣創(chuàng)建集合才能避免造成內(nèi)存泄漏你了解嗎

 更新時(shí)間:2021年09月29日 08:46:42   作者:Hollis Chuang  
內(nèi)存泄漏是指無(wú)用對(duì)象持續(xù)占有內(nèi)存或無(wú)用對(duì)象的內(nèi)存得不到及時(shí)釋放,從而造成內(nèi)存空間的浪費(fèi)稱(chēng)為內(nèi)存泄漏。長(zhǎng)生命周期的對(duì)象持有短生命周期對(duì)象的引用就很可能發(fā)生內(nèi)存泄漏,盡管短生命周期對(duì)象已經(jīng)不再需要,但是因?yàn)殚L(zhǎng)生命周期持有它的引用而導(dǎo)致不能被回收

由于Java語(yǔ)言的集合框架中(collections, 如list, map, set等)沒(méi)有提供任何簡(jiǎn)便的語(yǔ)法結(jié)構(gòu),這使得在建立常量集合時(shí)的工作非常繁索。每次建立時(shí)我們都要做:

1、定義一個(gè)空的集合類(lèi)變量

2、向這個(gè)結(jié)合類(lèi)中逐一添加元素

3、將集合做為參數(shù)傳遞給方法

例如,要將一個(gè)Set變量傳給一個(gè)方法:

Set users = new HashSet();
users.add("Hollis");
users.add("hollis");
users.add("HollisChuang");
users.add("hollis666");
transferUsers(users);

這樣的寫(xiě)法稍微有些復(fù)雜,有沒(méi)有簡(jiǎn)潔的方式呢?

雙括號(hào)語(yǔ)法初始化集合

其實(shí)有一個(gè)比較簡(jiǎn)潔的方式,那就是雙括號(hào)語(yǔ)法(double-brace syntax)建立并初始化一個(gè)新的集合:

public class DoubleBraceTest {
    public static void main(String[] args) {
        Set users = new HashSet() {{
            add("Hollis");
            add("hollis");
            add("HollisChuang");
            add("hollis666");
        }};
    }
}

同理,創(chuàng)建并初始化一個(gè)HashMap的語(yǔ)法如下:

Map<String,String> users = new HashMap<>() {{
    put("Hollis","Hollis");
    put("hollis","hollis");
    put("HollisChuang","HollisChuang");
}};

不只是Set、Map,jdk中的集合類(lèi)都可以用這種方式創(chuàng)建并初始化。

當(dāng)我們使用這種雙括號(hào)語(yǔ)法初始化集合類(lèi)的時(shí)候,在對(duì)Java文件進(jìn)行編譯時(shí),可以發(fā)現(xiàn)一個(gè)奇怪的現(xiàn)象,使用javac對(duì)DoubleBraceTest進(jìn)行編譯:

javac DoubleBraceTest.java

我們會(huì)發(fā)現(xiàn),得到兩個(gè)class文件:

DoubleBraceTest.class
DoubleBraceTest$1.class

有經(jīng)驗(yàn)的朋友可能一看到這兩個(gè)文件就會(huì)知道,這里面一定用到了匿名內(nèi)部類(lèi)。

沒(méi)錯(cuò),使用這個(gè)雙括號(hào)初始化的效果是創(chuàng)建匿名內(nèi)部類(lèi)。創(chuàng)建的類(lèi)有一個(gè)隱式的this指針指向外部類(lèi)。

不建議使用這種形式

首先,使用這種形式創(chuàng)建并初始化集合會(huì)導(dǎo)致很多內(nèi)部類(lèi)被創(chuàng)建。因?yàn)槊看问褂秒p大括號(hào)初始化時(shí),都會(huì)生成一個(gè)新類(lèi)。如這個(gè)例子:

Map hollis = new HashMap(){{
    put("firstName", "Hollis");
    put("lastName", "Chuang");
    put("contacts", new HashMap(){{
        put("0", new HashMap(){{
            put("blogs", "http://www.hollischuang.com");
        }});
        put("1", new HashMap(){{
            put("wechat", "hollischuang");
        }});
    }});
}};

這會(huì)使得很多內(nèi)部類(lèi)被創(chuàng)建出來(lái):

DoubleBraceTest$1$1$1.class
DoubleBraceTest$1$1$2.class
DoubleBraceTest$1$1.class
DoubleBraceTest$1.class
DoubleBraceTest.class

這些內(nèi)部類(lèi)被創(chuàng)建出來(lái),是需要被類(lèi)加載器加載的,這就帶來(lái)了一些額外的開(kāi)銷(xiāo)。

如果您使用上面的代碼在一個(gè)方法中創(chuàng)建并初始化一個(gè)map,并從方法返回該map,那么該方法的調(diào)用者可能會(huì)毫不知情地持有一個(gè)無(wú)法進(jìn)行垃圾收集的資源。

public Map getMap() {
    Map hollis = new HashMap(){{
        put("firstName", "Hollis");
        put("lastName", "Chuang");
        put("contacts", new HashMap(){{
            put("0", new HashMap(){{
                put("blogs", "http://www.hollischuang.com");
            }});
            put("1", new HashMap(){{
                put("wechat", "hollischuang");
            }});
        }});
    }};

    return hollis;
}

我們嘗試通過(guò)調(diào)用getMap得到這樣一個(gè)通過(guò)雙括號(hào)初始化出來(lái)的map

public class DoubleBraceTest {
    public static void main(String[] args) {
        DoubleBraceTest doubleBraceTest = new DoubleBraceTest();
        Map map = doubleBraceTest.getMap();
    }
}

返回的Map現(xiàn)在將包含一個(gè)對(duì)DoubleBraceTest的實(shí)例的引用。讀者可以嘗試這通過(guò)debug或者以下方式確認(rèn)這一事實(shí)。

Field field = map.getClass().getDeclaredField("this$0");
field.setAccessible(true);
System.out.println(field.get(map).getClass());

替代方案

很多人使用雙括號(hào)初始化集合,主要是因?yàn)樗容^方便,可以在定義集合的同時(shí)對(duì)他進(jìn)行初始化。

但其實(shí),目前已經(jīng)有很多方案可以做這個(gè)事情了,不需要再使用這種存在風(fēng)險(xiǎn)的方案。

使用Arrays工具類(lèi)

當(dāng)我們想要初始化一個(gè)List的時(shí)候,可以借助Arrays類(lèi),Arrays中提供了asList可以把一個(gè)數(shù)組轉(zhuǎn)換成List:

List<String> list2 = Arrays.asList("hollis ", "Hollis", "HollisChuang");

但是需要注意的是,asList 得到的只是一個(gè) Arrays 的內(nèi)部類(lèi),是一個(gè)原來(lái)數(shù)組的視圖 List,因此如果對(duì)它進(jìn)行增刪操作會(huì)報(bào)錯(cuò)。

使用Stream

Stream是Java中提供的新特性,他可以對(duì)傳入流內(nèi)部的元素進(jìn)行篩選、排序、聚合等中間操作(intermediate operate),最后由最終操作(terminal operation)得到前面處理的結(jié)果。

我們可以借助Stream來(lái)初始化集合:

List<String> list1 = Stream.of("hollis", "Hollis", "HollisChuang").collect(Collectors.toList());

使用第三方工具類(lèi)

很多第三方的集合工具類(lèi)可以實(shí)現(xiàn)這個(gè)功能,如Guava等:

ImmutableMap.of("k1", "v1", "k2", "v2");
ImmutableList.of("a", "b", "c", "d");

關(guān)于Guava和其中定義的不可變集合,我們?cè)诤竺鏁?huì)詳細(xì)介紹

Java 9內(nèi)置方法

其實(shí)在Java 9 中,在List、Map等集合類(lèi)中已經(jīng)內(nèi)置了初始化的方法,如List中包含了12個(gè)重載的of方法,就是來(lái)做這個(gè)事情的:

/**
 * Returns an unmodifiable list containing zero elements.
 *
 * See <a href="#unmodifiable" rel="external nofollow" >Unmodifiable Lists</a> for details.
 *
 * @param <E> the {@code List}'s element type
 * @return an empty {@code List}
 *
 * @since 9
 */
static <E> List<E> of() {
    return ImmutableCollections.emptyList();
}

static <E> List<E> of(E e1) {
    return new ImmutableCollections.List12<>(e1);
}

static <E> List<E> of(E... elements) {
    switch (elements.length) { // implicit null check of elements
        case 0:
            return ImmutableCollections.emptyList();
        case 1:
            return new ImmutableCollections.List12<>(elements[0]);
        case 2:
            return new ImmutableCollections.List12<>(elements[0], elements[1]);
        default:
            return new ImmutableCollections.ListN<>(elements);
    }
}

到此這篇關(guān)于Java怎樣創(chuàng)建集合才能避免造成內(nèi)存泄漏你了解嗎的文章就介紹到這了,更多相關(guān)Java 集合內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

最新評(píng)論