All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.landawn.abacus.util.BiMap Maven / Gradle / Ivy

/*
 * Copyright (c) 2015, Haiyang Li.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.landawn.abacus.util;

import java.util.AbstractSet;
import java.util.HashMap;
import java.util.Iterator;
import java.util.Map;
import java.util.Set;

import com.landawn.abacus.annotation.Internal;
import com.landawn.abacus.util.function.Supplier;

/**
 * A BiMap (or "bidirectional map") is a map that preserves the uniqueness of its values as well as that of its keys.
 * This constraint enables BiMaps to support an "inverse view", which is another BiMap containing the same entries as this BiMap but with reversed keys and values.
 *
 * @author Haiyang Li
 * @param  the key type
 * @param  the value type
 * @since 0.8
 */
public final class BiMap implements Map {
    /**
     * The maximum capacity, used if a higher value is implicitly specified by either of the constructors with
     * arguments. MUST be a power of two <= 1<<30.
     */
    static final int MAXIMUM_CAPACITY = 1 << 30;

    /**
     * The default initial capacity - MUST be a power of two.
     */
    static final int DEFAULT_INITIAL_CAPACITY = 1 << 4; // aka 16

    /**
     * The load factor used when none specified in constructor.
     */
    static final float DEFAULT_LOAD_FACTOR = 0.75f;

    final Supplier> keyMapSupplier;

    final Supplier> valueMapSupplier;

    final Map keyMap;

    final Map valueMap;

    private transient BiMap inverse;

    public BiMap() {
        this(DEFAULT_INITIAL_CAPACITY);
    }

    public BiMap(int initialCapacity) {
        this(initialCapacity, DEFAULT_LOAD_FACTOR);
    }

    @SuppressWarnings("deprecation")
    public BiMap(int initialCapacity, float loadFactor) {
        this(new HashMap(N.initHashCapacity(initialCapacity), loadFactor), new HashMap(N.initHashCapacity(initialCapacity), loadFactor));
    }

    @SuppressWarnings("rawtypes")
    public BiMap(final Class keyMapType, final Class valueMapType) {
        this(Maps.mapType2Supplier(keyMapType), Maps.mapType2Supplier(valueMapType));
    }

    public BiMap(final Supplier> keyMapSupplier, final Supplier> valueMapSupplier) {
        this.keyMapSupplier = keyMapSupplier;
        this.valueMapSupplier = valueMapSupplier;
        this.keyMap = keyMapSupplier.get();
        this.valueMap = valueMapSupplier.get();
    }

    @Internal
    BiMap(final Map keyMap, final Map valueMap) {
        this.keyMapSupplier = Maps.mapType2Supplier(keyMap.getClass());
        this.valueMapSupplier = Maps.mapType2Supplier(valueMap.getClass());
        this.keyMap = keyMap;
        this.valueMap = valueMap;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @return
     */
    public static  BiMap of(final K k1, final V v1) {
        final BiMap map = new BiMap<>(1);

        map.put(k1, v1);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @param k2
     * @param v2
     * @return
     */
    public static  BiMap of(final K k1, final V v1, final K k2, final V v2) {
        final BiMap map = new BiMap<>(2);

        map.put(k1, v1);
        map.put(k2, v2);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @param k2
     * @param v2
     * @param k3
     * @param v3
     * @return
     */
    public static  BiMap of(final K k1, final V v1, final K k2, final V v2, final K k3, final V v3) {
        final BiMap map = new BiMap<>(3);

        map.put(k1, v1);
        map.put(k2, v2);
        map.put(k3, v3);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @param k2
     * @param v2
     * @param k3
     * @param v3
     * @param k4
     * @param v4
     * @return
     */
    public static  BiMap of(final K k1, final V v1, final K k2, final V v2, final K k3, final V v3, final K k4, final V v4) {
        final BiMap map = new BiMap<>(4);

        map.put(k1, v1);
        map.put(k2, v2);
        map.put(k3, v3);
        map.put(k4, v4);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @param k2
     * @param v2
     * @param k3
     * @param v3
     * @param k4
     * @param v4
     * @param k5
     * @param v5
     * @return
     */
    public static  BiMap of(final K k1, final V v1, final K k2, final V v2, final K k3, final V v3, final K k4, final V v4, final K k5,
            final V v5) {
        final BiMap map = new BiMap<>(5);

        map.put(k1, v1);
        map.put(k2, v2);
        map.put(k3, v3);
        map.put(k4, v4);
        map.put(k5, v5);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @param k2
     * @param v2
     * @param k3
     * @param v3
     * @param k4
     * @param v4
     * @param k5
     * @param v5
     * @param k6
     * @param v6
     * @return
     */
    public static  BiMap of(final K k1, final V v1, final K k2, final V v2, final K k3, final V v3, final K k4, final V v4, final K k5, final V v5,
            final K k6, final V v6) {
        final BiMap map = new BiMap<>(6);

        map.put(k1, v1);
        map.put(k2, v2);
        map.put(k3, v3);
        map.put(k4, v4);
        map.put(k5, v5);
        map.put(k6, v6);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param 
     * @param 
     * @param k1
     * @param v1
     * @param k2
     * @param v2
     * @param k3
     * @param v3
     * @param k4
     * @param v4
     * @param k5
     * @param v5
     * @param k6
     * @param v6
     * @param k7
     * @param v7
     * @return
     */
    public static  BiMap of(final K k1, final V v1, final K k2, final V v2, final K k3, final V v3, final K k4, final V v4, final K k5, final V v5,
            final K k6, final V v6, final K k7, final V v7) {
        final BiMap map = new BiMap<>(7);

        map.put(k1, v1);
        map.put(k2, v2);
        map.put(k3, v3);
        map.put(k4, v4);
        map.put(k5, v5);
        map.put(k6, v6);
        map.put(k7, v7);

        return map;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param map
     * @return
     */
    public static  BiMap copyOf(final Map map) {
        final BiMap biMap = new BiMap<>(Maps.newTargetMap(map), Maps.newOrderingMap(map));

        biMap.putAll(map);

        return biMap;
    }

    /**
     *
     * @param  the key type
     * @param  the value type
     * @param map
     * @return
     * @deprecated replaced by {@code copyOf}
     */
    @Deprecated
    public static  BiMap from(final Map map) {
        return copyOf(map);
    }

    /**
     *
     * @param key
     * @return
     */
    @Override
    public V get(Object key) {
        return keyMap.get(key);
    }

    /**
     * Gets the by value.
     *
     * @param value
     * @return
     */
    public K getByValue(Object value) {
        return valueMap.get(value);
    }

    /**
     *
     * @param key
     * @param value
     * @return
     * @throws IllegalArgumentException if the given value is already bound to a
     *     different key in this bimap. The bimap will remain unmodified in this
     *     event. To avoid this exception, call {@link #forcePut} instead.
     */
    @Override
    public V put(final K key, final V value) {
        return put(key, value, false);
    }

    /**
     * An alternate form of {@code put} that silently removes any existing entry
     * with the value {@code value} before proceeding with the {@link #put}
     * operation. If the bimap previously contained the provided key-value
     * mapping, this method has no effect.
     *
     * 

Note that a successful call to this method could cause the size of the * bimap to increase by one, stay the same, or even decrease by one. * *

Warning: If an existing entry with this value is removed, the key * for that entry is discarded and not returned. * * @param key the key with which the specified value is to be associated * @param value the value to be associated with the specified key * @return * be {@code null}, or {@code null} if there was no previous entry */ public V forcePut(final K key, final V value) { return put(key, value, true); } /** * * @param key * @param value * @param isForce * @return */ private V put(final K key, final V value, final boolean isForce) { if ((key == null) || (value == null)) { throw new NullPointerException("key or value can't be null"); } else if (isForce == false && valueMap.containsKey(value)) { throw new IllegalArgumentException("Value already exists: " + value); } V v = keyMap.remove(key); if (v != null) { valueMap.remove(v); } K k = valueMap.remove(value); if (k != null) { keyMap.remove(k); } keyMap.put(key, value); valueMap.put(value, key); return v; } /** *

Warning: the results of calling this method may vary depending on * the iteration order of {@code map}. * * @param m * @throws IllegalArgumentException if an attempt to {@code put} any * entry fails. Note that some map entries may have been added to the * bimap before the exception was thrown. */ @Override public void putAll(Map m) { for (Map.Entry e : m.entrySet()) { put(e.getKey(), e.getValue()); } } /** * * @param key * @return */ @Override public V remove(Object key) { V value = keyMap.remove(key); if (value != null) { valueMap.remove(value); } return value; } /** * Removes the by value. * * @param value * @return */ public K removeByValue(Object value) { K key = valueMap.remove(value); if (key != null) { keyMap.remove(key); } return key; } /** * * @param key * @return */ @Override public boolean containsKey(Object key) { return keyMap.containsKey(key); } /** * * @param value * @return */ @Override public boolean containsValue(Object value) { return valueMap.containsKey(value); } /** * Returns an immutable key set. * * @return */ @Override public ImmutableSet keySet() { return ImmutableSet.of(keyMap.keySet()); } /** * Returns an immutable value set. * * @return */ @Override public ImmutableSet values() { return ImmutableSet.of(valueMap.keySet()); } /** * Returns an immutable Set of Immutable entry. * * @return */ @Override public Set> entrySet() { return new AbstractSet>() { @Override public Iterator> iterator() { return new ObjIterator>() { private final Iterator> keyValueEntryIter = keyMap.entrySet().iterator(); @Override public boolean hasNext() { return keyValueEntryIter.hasNext(); } @Override public Map.Entry next() { final Map.Entry entry = keyValueEntryIter.next(); return new Map.Entry() { @Override public K getKey() { return entry.getKey(); } @Override public V getValue() { return entry.getValue(); } @Override public V setValue(V value) { // if (N.equals(entry.getValue(), value)) { // return entry.getValue(); // } // // // if (valueMap.containsKey(value)) { // // throw new IllegalStateException("Value: " + N.toString(value) + " already existed."); // // } // // valueMap.remove(entry.getValue()); // valueMap.put(value, entry.getKey()); // return entry.setValue(value); throw new UnsupportedOperationException(); } }; } }; } @Override public int size() { return keyMap.size(); } }; } /** * Returns the inverse view of this BiMap, which maps each of this bimap's values to its associated key. * The two BiMaps are backed by the same data; any changes to one will appear in the other. * * @return */ public BiMap inversed() { return (inverse == null) ? inverse = new BiMap<>(valueMap, keyMap) : inverse; } public BiMap copy() { final BiMap copy = new BiMap<>(keyMapSupplier, valueMapSupplier); copy.putAll(keyMap); return copy; } /** * Clear. */ @Override public void clear() { keyMap.clear(); valueMap.clear(); } /** * Checks if is empty. * * @return true, if is empty */ @Override public boolean isEmpty() { return keyMap.isEmpty(); } @Override public int size() { return keyMap.size(); } // public Stream> stream() { // return Stream.of(keyMap.entrySet()); // } @Override public int hashCode() { return keyMap.hashCode(); } /** * * @param obj * @return */ @SuppressWarnings("unchecked") @Override public boolean equals(Object obj) { return obj == this || (obj instanceof BiMap && keyMap.equals(((BiMap) obj).keyMap)); } @Override public String toString() { return keyMap.toString(); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy