The interesting case is this one: say there are mappings A->1 and B->2. What happens once you insert C->2 ?
Will you remove B->2 to ensure uniqueness of keys and values ? (resulting in A->1 and C->2)
or not, resulting in A->1, B->2 and C->2. Please note that in the latter case, there's no well-defined lookup for value "2". Is it B or is it C ?
So what I needed was uniqueness of both keys and values, in all cases, using generics. Here it goes:
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Set;
/**
* One possible implementation of a Bi-directional Map.
*
* Note: there are several variations of BiMap, depending on the behaviour of put(k,v)... Is it allowed to have more
* than one k for one value ?? Modify
*
* @author apodehl
*
* @param
* @param
*/
public class BiMap implements Map
{
Map keyVal; // maps key->value
Map valKey; // maps value->key
public BiMap()
{
keyVal = new LinkedHashMap();
valKey = new LinkedHashMap();
}
@Override
public V put(K key, V val)
{
// --- this implementation allows one-to-one ONLY !! ---
if (valKey.containsKey(val)) {
keyVal.remove(valKey.get(val));
}
if (keyVal.containsKey(key)) {
valKey.remove(keyVal.get(key));
}
// --- remove above if that's not what you want ---
//
valKey.put(val, key);
return keyVal.put(key, val);
}
@Override
public void putAll(Map< ? extends K, ? extends V> m)
{
for (Entry< ? extends K, ? extends V> e : m.entrySet()) {
put(e.getKey(), e.getValue());
}
}
/**
* Type-safe get.
*
* @param key
* @return
*/
public K getKey(V val)
{
return valKey.get(val);
}
/**
* Type-safe get.
*
* @param key
* @return
*/
public V getVal(K key)
{
return keyVal.get(key);
}
@Override
// unfortunately Map interface isn't type-safe
public V get(Object key)
{
return getVal((K) key);
}
/**
* Type-safe contains.
*
* @param key
* @return
*/
public boolean contains(K key)
{
if (keyVal == null) return false;
return keyVal.containsKey(key);
}
/**
* Type-safe containsValue.
*
* @param key
* @return
*/
public boolean containsVal(V val)
{
if (valKey == null) return false;
return valKey.containsKey(val);
}
// unfortunately Map interface isn't type safe here ..
@Override
public boolean containsKey(Object key)
{
return contains((K) key);
}
// unfortunately Map interface isn't type safe here ..
@Override
public boolean containsValue(Object value)
{
return containsVal((V) value);
}
@Override
public Set< java.util.Map.Entry> entrySet()
{
return keyVal.entrySet();
}
@Override
public boolean isEmpty()
{
return keyVal.isEmpty();
}
@Override
public int size()
{
return keyVal.size();
}
@Override
public Collection values()
{
return keyVal.values();
}
public V removeByKey(K key)
{
V val = keyVal.remove(key);
valKey.remove(val);
return val;
}
public K removeByVal(V val)
{
K key = valKey.remove(val);
keyVal.remove(key);
return key;
}
// unfortunately Map interface isn't type-safe
@Override
public V remove(Object key)
{
return removeByKey((K) key);
}
@Override
public Set keySet()
{
return keyVal.keySet();
}
@Override
public void clear()
{
keyVal.clear();
valKey.clear();
}
public String toString()
{
String s = "BiMap:\n";
s += " key->value: " + keyVal.toString();
s += "\n";
s += " value->key: " + valKey.toString();
return s;
}
public static void main(String[] args)
{
BiMap biMap = new BiMap();
biMap.put("A", 1);
biMap.put("B", 2);
System.out.println(biMap); // {A=1, B=2}
biMap.put("C", 2);
System.out.println(biMap); // {A=1, C=2}
biMap.removeByVal(1);
System.out.println(biMap); // {C=2}
}
}
Great! This is exactly what I needed! Thanks for sharing it!
ReplyDeleteI made some changes:
ReplyDeleteimport java.util.Collection;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Set;
public class BiMap implements Map {
private Map keyVal; // maps key->value
private Map valKey; // maps value->key
@Override
public int size() {
return keyVal.size();
}
@Override
public boolean isEmpty() {
return keyVal.isEmpty();
}
@Override
public boolean containsKey(Object key) {
return contains((K) key);
}
/**
* Type-safe contains.
*
* @param key
* @return
*/
public boolean contains(K key) {
if (keyVal == null) {
return false;
}
return keyVal.containsKey(key);
}
@Override
public boolean containsValue(Object value) {
return containsVal((V) value);
}
/**
* Type-safe containsValue.
*
* @param key
* @return
*/
public boolean containsVal(V value) {
if (valKey == null) {
return false;
}
return valKey.containsKey(value);
}
@Override
public V get(Object key) {
return getVal((K) key);
}
/**
* Type-safe get.
*
* @param key
* @return
*/
public V getVal(K key) {
return keyVal.get(key);
}
@Override
public V put(K key, V value) {
/* this implementation allows one-to-one ONLY */
if (valKey.containsKey(value)) {
keyVal.remove(valKey.get(value));
}
if (keyVal.containsKey(key)) {
valKey.remove(keyVal.get(key));
}
/* remove above if that's not what you want */
valKey.put(value, key);
return keyVal.put(key, value);
}
@Override
public V remove(Object key) {
return removeByKey((K) key);
}
public V removeByKey(K key) {
V value = keyVal.remove(key);
valKey.remove(value);
return value;
}
@Override
public void putAll(Map m) {
for (Entry e : m.entrySet()) {
put(e.getKey(), e.getValue());
}
}
@Override
public void clear() {
keyVal.clear();
valKey.clear();
}
@Override
public Set keySet() {
return keyVal.keySet();
}
@Override
public Collection values() {
return keyVal.values();
}
@Override
public Set> entrySet() {
return keyVal.entrySet();
}
}
Hi Diego, thanks for your feedback. What exactly did you change and why ?
ReplyDelete