當前位置:
首頁 > 知識 > Map 大家族的那點事兒( 2 ) :AbstractMap

Map 大家族的那點事兒( 2 ) :AbstractMap

(點擊

上方公眾號

,可快速關注)




來源:SylvanasSun』s Blog ,


sylvanassun.github.io/2018/03/16/2018-03-16-map_family/




AbstractMap



AbstractMap是一個抽象類,它是Map介面的一個骨架實現,最小化實現了此介面提供的抽象函數。在Java的Collection框架中基本都遵循了這一規定,骨架實現在介面與實現類之間構建了一層抽象,其目的是為了復用一些比較通用的函數以及方便擴展,例如List介面擁有骨架實現AbstractList、Set介面擁有骨架實現AbstractSet等。




下面我們按照不同的操作類型來看看AbstractMap都實現了什麼,首先是查詢操作:





package java.util;


import java.util.Map.Entry;


public abstract class AbstractMap<K,V> implements Map<K,V> {


 


    protected AbstractMap() {


    }


    // Query Operations


    public int size() {


        return entrySet().size();


    }


    // 鍵值對的集合視圖留給具體的實現類實現


    public abstract Set<Entry<K,V>> entrySet();


    public boolean isEmpty() {


        return size() == 0;


    }


    /**


     * 遍歷entrySet,然後逐個進行比較。


     */


    public boolean containsValue(Object value) {


        Iterator<Entry<K,V>> i = entrySet().iterator();


        if (value==null) {


            while (i.hasNext()) {


                Entry<K,V> e = i.next();


                if (e.getValue()==null)


                    return true;


            }


        } else {


            while (i.hasNext()) {


                Entry<K,V> e = i.next();


                if (value.equals(e.getValue()))


                    return true;


            }


        }


        return false;


    }


    /**


     * 跟containsValue()同理,只不過比較的是key。


     */


    public boolean containsKey(Object key) {


        Iterator<Map.Entry<K,V>> i = entrySet().iterator();


        if (key==null) {


            while (i.hasNext()) {


                Entry<K,V> e = i.next();


                if (e.getKey()==null)


                    return true;


            }


        } else {


            while (i.hasNext()) {


                Entry<K,V> e = i.next();


                if (key.equals(e.getKey()))


                    return true;


            }


        }


        return false;


    }


    /**


     * 遍歷entrySet,然後根據key取出關聯的value。


     */


    public V get(Object key) {


        Iterator<Entry<K,V>> i = entrySet().iterator();


        if (key==null) {


            while (i.hasNext()) {


                Entry<K,V> e = i.next();


                if (e.getKey()==null)


                    return e.getValue();


            }


        } else {


            while (i.hasNext()) {


                Entry<K,V> e = i.next();


                if (key.equals(e.getKey()))


                    return e.getValue();


            }


        }


        return null;


    }


}




可以發現這些操作都是依賴於函數entrySet()的,它返回了一個鍵值對的集合視圖,由於不同的實現子類的Entry實現可能也是不同的,所以一般是在內部實現一個繼承於AbstractSet且泛型為Map.Entry的內部類作為EntrySet,接下來是修改操作與批量操作:





// Modification Operations


/**


 * 沒有提供實現,子類必須重寫該方法,否則調用put()會拋出異常。


 */


public V put(K key, V value) {


    throw new UnsupportedOperationException();


}


/**


 * 遍歷entrySet,先找到目標的entry,然後刪除。


 *(還記得之前說過的嗎,集合視圖中的操作也會影響到實際數據)


 */


public V remove(Object key) {


    Iterator<Entry<K,V>> i = entrySet().iterator();


    Entry<K,V> correctEntry = null;


    if (key==null) {


        while (correctEntry==null && i.hasNext()) {


            Entry<K,V> e = i.next();


            if (e.getKey()==null)


                correctEntry = e;


        }


    } else {


        while (correctEntry==null && i.hasNext()) {


            Entry<K,V> e = i.next();


            if (key.equals(e.getKey()))


                correctEntry = e;


        }


    }


    V oldValue = null;


    if (correctEntry !=null) {


        oldValue = correctEntry.getValue();


        i.remove();


    }


    return oldValue;


}


// Bulk Operations


/**


 * 遍歷參數m,然後將每一個鍵值對put到該Map中。


 */


public void putAll(Map<? extends K, ? extends V> m) {


    for (Map.Entry<? extends K, ? extends V> e : m.entrySet())


        put(e.getKey(), e.getValue());


}


/**


 * 清空entrySet等價於清空該Map。


 */


public void clear() {


    entrySet().clear();


}




AbstractMap並沒有實現put()函數,這樣做是為了考慮到也許會有不可修改的Map實現子類繼承它,而對於一個可修改的Map實現子類則必須重寫put()函數。




AbstractMap沒有提供entrySet()的實現,但是卻提供了keySet()與values()集合視圖的默認實現,它們都是依賴於entrySet()返回的集合視圖實現的,源碼如下:





/**


 * keySet和values是lazy的,它們只會在第一次請求視圖時進行初始化,


 * 而且它們是無狀態的,所以只需要一個實例(初始化一次)。


 */


transient Set<K>        keySet;


transient Collection<V> values;


/**


 * 返回一個AbstractSet的子類,可以發現它的行為都委託給了entrySet返回的集合視圖


 * 與當前的AbstractMap實例,所以說它自身是無狀態的。


 */


public Set<K> keySet() {


    Set<K> ks = keySet;


    if (ks == null) {


        ks = new AbstractSet<K>() {


            public Iterator<K> iterator() {


                return new Iterator<K>() {


                    private Iterator<Entry<K,V>> i = entrySet().iterator();


                    public boolean hasNext() {


                        return i.hasNext();


                    }


                    public K next() {


                        return i.next().getKey();


                    }


                    public void remove() {


                        i.remove();


                    }


                };


            }


            public int size() {


                return AbstractMap.this.size();


            }


            public boolean isEmpty() {


                return AbstractMap.this.isEmpty();


            }


            public void clear() {


                AbstractMap.this.clear();


            }


            public boolean contains(Object k) {


                return AbstractMap.this.containsKey(k);


            }


        };


        keySet = ks;


    }


    return ks;


}


/**


 * 與keySet()基本一致,唯一的區別就是返回的是AbstractCollection的子類,


 * 主要是因為value不需要保持互異性。


 */


public Collection<V> values() {


    Collection<V> vals = values;


    if (vals == null) {


        vals = new AbstractCollection<V>() {


            public Iterator<V> iterator() {


                return new Iterator<V>() {


                    private Iterator<Entry<K,V>> i = entrySet().iterator();


                    public boolean hasNext() {


                        return i.hasNext();


                    }


                    public V next() {


                        return i.next().getValue();


                    }


                    public void remove() {


                        i.remove();


                    }


                };


            }


            public int size() {


                return AbstractMap.this.size();


            }


            public boolean isEmpty() {


                return AbstractMap.this.isEmpty();


            }


            public void clear() {


                AbstractMap.this.clear();


            }


            public boolean contains(Object v) {


                return AbstractMap.this.containsValue(v);


            }


        };


        values = vals;


    }


    return vals;


}




它還提供了兩個Entry的實現類:SimpleEntry與SimpleImmutableEntry,這兩個類的實現非常簡單,區別也只是前者是可變的,而後者是不可變的。





private static boolean eq(Object o1, Object o2) {


    return o1 == null ? o2 == null : o1.equals(o2);


}


public static class SimpleEntry<K,V>


    implements Entry<K,V>, java.io.Serializable


{


    private static final long serialVersionUID = -8499721149061103585L;


    private final K key;


    private V value;


    public SimpleEntry(K key, V value) {


        this.key   = key;


        this.value = value;


    }


    public SimpleEntry(Entry<? extends K, ? extends V> entry) {


        this.key   = entry.getKey();


        this.value = entry.getValue();


    }


    public K getKey() {


        return key;


    }


    public V getValue() {


        return value;


    }


    public V setValue(V value) {


        V oldValue = this.value;


        this.value = value;


        return oldValue;


    }


    public boolean equals(Object o) {


        if (!(o instanceof Map.Entry))


            return false;


        Map.Entry<?,?> e = (Map.Entry<?,?>)o;


        return eq(key, e.getKey()) && eq(value, e.getValue());


    }


    public int hashCode() {


        return (key   == null ? 0 :   key.hashCode()) ^


               (value == null ? 0 : value.hashCode());


    }


    public String toString() {


        return key + "=" + value;


    }


}


/**


 * 它與SimpleEntry的區別在於它是不可變的,value被final修飾,並且不支持setValue()。


 */


public static class SimpleImmutableEntry<K,V>


    implements Entry<K,V>, java.io.Serializable


{


    private static final long serialVersionUID = 7138329143949025153L;


    private final K key;


    private final V value;


    public SimpleImmutableEntry(K key, V value) {


        this.key   = key;


        this.value = value;


    }


    public SimpleImmutableEntry(Entry<? extends K, ? extends V> entry) {


        this.key   = entry.getKey();


        this.value = entry.getValue();


    }


    public K getKey() {


        return key;


    }


    public V getValue() {


        return value;


    }


    public V setValue(V value) {


        throw new UnsupportedOperationException();


    }


    public boolean equals(Object o) {


        if (!(o instanceof Map.Entry))


            return false;


        Map.Entry<?,?> e = (Map.Entry<?,?>)o;


        return eq(key, e.getKey()) && eq(value, e.getValue());


    }


    public int hashCode() {


        return (key   == null ? 0 :   key.hashCode()) ^


               (value == null ? 0 : value.hashCode());


    }


    public String toString() {


        return key + "=" + value;


    }


}




我們通過閱讀上述的源碼不難發現,AbstractMap實現的操作都依賴於entrySet()所返回的集合視圖。剩下的函數就沒什麼好說的了,有興趣的話可以自己去看看。




【關於投稿】




如果大家有原創好文投稿,請直接給公號發送留言。




① 留言格式:


【投稿】+《 文章標題》+ 文章鏈接

② 示例:


【投稿】《不要自稱是程序員,我十多年的 IT 職場總結》:http://blog.jobbole.com/94148/

③ 最後請附上您的個人簡介哈~






看完本文有收穫?請轉發分享給更多人


關注「ImportNew」,提升Java技能


喜歡這篇文章嗎?立刻分享出去讓更多人知道吧!

本站內容充實豐富,博大精深,小編精選每日熱門資訊,隨時更新,點擊「搶先收到最新資訊」瀏覽吧!


請您繼續閱讀更多來自 ImportNew 的精彩文章:

使用 JITWatch 查看 JVM 的 JIT 編譯代碼
SpringBoot | 第九章:Mybatis-plus 的集成和使用

TAG:ImportNew |