歡迎光臨
每天分享高質量文章

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 implements Map {

 

    protected AbstractMap() {

    }

    // Query Operations

    public int size() {

        return entrySet().size();

    }

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

    public abstract Set> entrySet();

    public boolean isEmpty() {

        return size() == 0;

    }

    /**

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

     */

    public boolean containsValue(Object value) {

        Iterator> i = entrySet().iterator();

        if (value==null) {

            while (i.hasNext()) {

                Entry e = i.next();

                if (e.getValue()==null)

                    return true;

            }

        } else {

            while (i.hasNext()) {

                Entry e = i.next();

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

                    return true;

            }

        }

        return false;

    }

    /**

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

     */

    public boolean containsKey(Object key) {

        Iterator> i = entrySet().iterator();

        if (key==null) {

            while (i.hasNext()) {

                Entry e = i.next();

                if (e.getKey()==null)

                    return true;

            }

        } else {

            while (i.hasNext()) {

                Entry e = i.next();

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

                    return true;

            }

        }

        return false;

    }

    /**

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

     */

    public V get(Object key) {

        Iterator> i = entrySet().iterator();

        if (key==null) {

            while (i.hasNext()) {

                Entry e = i.next();

                if (e.getKey()==null)

                    return e.getValue();

            }

        } else {

            while (i.hasNext()) {

                Entry 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> i = entrySet().iterator();

    Entry correctEntry = null;

    if (key==null) {

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

            Entry e = i.next();

            if (e.getKey()==null)

                correctEntry = e;

        }

    } else {

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

            Entry 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        keySet;

transient Collection values;

/**

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

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

 */

public Set keySet() {

    Set ks = keySet;

    if (ks == null) {

        ks = new AbstractSet() {

            public Iterator iterator() {

                return new Iterator() {

                    private Iterator> 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 values() {

    Collection vals = values;

    if (vals == null) {

        vals = new AbstractCollection() {

            public Iterator iterator() {

                return new Iterator() {

                    private Iterator> 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

    implements Entry, 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

    implements Entry, 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技能

贊(0)

分享創造快樂