/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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 org.apache.cassandra.utils;

import java.util.Collection;
import java.util.HashMap;
import java.util.Map;
import java.util.Set;

import com.google.common.collect.HashMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Multimaps;

A variant of BiMap which does not enforce uniqueness of values. This means the inverse is a Multimap. (But the "forward" view is not a multimap; keys may only each have one value.)
Type parameters:
  • <K> –
  • <V> –
/** * * A variant of BiMap which does not enforce uniqueness of values. This means the inverse * is a Multimap. (But the "forward" view is not a multimap; keys may only each have one value.) * * @param <K> * @param <V> */
public class BiMultiValMap<K, V> implements Map<K, V> { protected final Map<K, V> forwardMap; protected final Multimap<V, K> reverseMap; public BiMultiValMap() { this.forwardMap = new HashMap<K, V>(); this.reverseMap = HashMultimap.<V, K>create(); } protected BiMultiValMap(Map<K, V> forwardMap, Multimap<V, K> reverseMap) { this.forwardMap = forwardMap; this.reverseMap = reverseMap; } public BiMultiValMap(BiMultiValMap<K, V> map) { this(); forwardMap.putAll(map); reverseMap.putAll(map.inverse()); } public Multimap<V, K> inverse() { return Multimaps.unmodifiableMultimap(reverseMap); } public void clear() { forwardMap.clear(); reverseMap.clear(); } public boolean containsKey(Object key) { return forwardMap.containsKey(key); } public boolean containsValue(Object value) { return reverseMap.containsKey(value); } public Set<Map.Entry<K, V>> entrySet() { return forwardMap.entrySet(); } public V get(Object key) { return forwardMap.get(key); } public boolean isEmpty() { return forwardMap.isEmpty(); } public Set<K> keySet() { return forwardMap.keySet(); } public V put(K key, V value) { V oldVal = forwardMap.put(key, value); if (oldVal != null) reverseMap.remove(oldVal, key); reverseMap.put(value, key); return oldVal; } public void putAll(Map<? extends K, ? extends V> m) { for (Map.Entry<? extends K, ? extends V> entry : m.entrySet()) put(entry.getKey(), entry.getValue()); } public V remove(Object key) { V oldVal = forwardMap.remove(key); reverseMap.remove(oldVal, key); return oldVal; } public Collection<K> removeValue(V value) { Collection<K> keys = reverseMap.removeAll(value); for (K key : keys) forwardMap.remove(key); return keys; } public int size() { return forwardMap.size(); } public Collection<V> values() { return reverseMap.keys(); } public Collection<V> valueSet() { return reverseMap.keySet(); } }