001/* 002 * Licensed to the Apache Software Foundation (ASF) under one or more 003 * contributor license agreements. See the NOTICE file distributed with 004 * this work for additional information regarding copyright ownership. 005 * The ASF licenses this file to You under the Apache License, Version 2.0 006 * (the "License"); you may not use this file except in compliance with 007 * the License. You may obtain a copy of the License at 008 * 009 * http://www.apache.org/licenses/LICENSE-2.0 010 * 011 * Unless required by applicable law or agreed to in writing, software 012 * distributed under the License is distributed on an "AS IS" BASIS, 013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 014 * See the License for the specific language governing permissions and 015 * limitations under the License. 016 */ 017package org.apache.commons.collections4.bidimap; 018 019import java.io.IOException; 020import java.io.ObjectInputStream; 021import java.io.ObjectOutputStream; 022import java.io.Serializable; 023import java.util.ArrayList; 024import java.util.Comparator; 025import java.util.Iterator; 026import java.util.ListIterator; 027import java.util.Map; 028import java.util.SortedMap; 029import java.util.TreeMap; 030 031import org.apache.commons.collections4.BidiMap; 032import org.apache.commons.collections4.OrderedBidiMap; 033import org.apache.commons.collections4.OrderedMap; 034import org.apache.commons.collections4.OrderedMapIterator; 035import org.apache.commons.collections4.ResettableIterator; 036import org.apache.commons.collections4.SortedBidiMap; 037import org.apache.commons.collections4.map.AbstractSortedMapDecorator; 038 039/** 040 * Implements {@link BidiMap} with two {@link TreeMap} instances. 041 * <p> 042 * The setValue() method on iterators will succeed only if the new value being set is 043 * not already in the bidi map. 044 * </p> 045 * <p> 046 * When considering whether to use this class, the {@link TreeBidiMap} class should 047 * also be considered. It implements the interface using a dedicated design, and does 048 * not store each object twice, which can save on memory use. 049 * </p> 050 * <p> 051 * NOTE: From Commons Collections 3.1, all subclasses will use {@link TreeMap} 052 * and the flawed {@code createMap} method is ignored. 053 * </p> 054 * 055 * @param <K> the type of the keys in this map 056 * @param <V> the type of the values in this map 057 * @since 3.0 058 */ 059public class DualTreeBidiMap<K, V> extends AbstractDualBidiMap<K, V> 060 implements SortedBidiMap<K, V>, Serializable { 061 062 /** 063 * Inner class MapIterator. 064 * 065 * @param <K> the type of the keys. 066 * @param <V> the type of the values. 067 */ 068 protected static class BidiOrderedMapIterator<K, V> implements OrderedMapIterator<K, V>, ResettableIterator<K> { 069 070 /** The parent map */ 071 private final AbstractDualBidiMap<K, V> parent; 072 073 /** The iterator being decorated */ 074 private ListIterator<Map.Entry<K, V>> iterator; 075 076 /** The last returned entry */ 077 private Map.Entry<K, V> last; 078 079 /** 080 * Constructs a new instance. 081 * @param parent the parent map 082 */ 083 protected BidiOrderedMapIterator(final AbstractDualBidiMap<K, V> parent) { 084 this.parent = parent; 085 iterator = new ArrayList<>(parent.entrySet()).listIterator(); 086 } 087 088 @Override 089 public K getKey() { 090 if (last == null) { 091 throw new IllegalStateException( 092 "Iterator getKey() can only be called after next() and before remove()"); 093 } 094 return last.getKey(); 095 } 096 097 @Override 098 public V getValue() { 099 if (last == null) { 100 throw new IllegalStateException( 101 "Iterator getValue() can only be called after next() and before remove()"); 102 } 103 return last.getValue(); 104 } 105 106 @Override 107 public boolean hasNext() { 108 return iterator.hasNext(); 109 } 110 111 @Override 112 public boolean hasPrevious() { 113 return iterator.hasPrevious(); 114 } 115 116 @Override 117 public K next() { 118 last = iterator.next(); 119 return last.getKey(); 120 } 121 122 @Override 123 public K previous() { 124 last = iterator.previous(); 125 return last.getKey(); 126 } 127 128 @Override 129 public void remove() { 130 iterator.remove(); 131 parent.remove(last.getKey()); 132 last = null; 133 } 134 135 @Override 136 public void reset() { 137 iterator = new ArrayList<>(parent.entrySet()).listIterator(); 138 last = null; 139 } 140 141 @Override 142 public V setValue(final V value) { 143 if (last == null) { 144 throw new IllegalStateException( 145 "Iterator setValue() can only be called after next() and before remove()"); 146 } 147 if (parent.reverseMap.containsKey(value) && 148 parent.reverseMap.get(value) != last.getKey()) { 149 throw new IllegalArgumentException( 150 "Cannot use setValue() when the object being set is already in the map"); 151 } 152 final V oldValue = parent.put(last.getKey(), value); 153 // Map.Entry specifies that the behavior is undefined when the backing map 154 // has been modified (as we did with the put), so we also set the value 155 last.setValue(value); 156 return oldValue; 157 } 158 159 @Override 160 public String toString() { 161 if (last != null) { 162 return "MapIterator[" + getKey() + "=" + getValue() + "]"; 163 } 164 return "MapIterator[]"; 165 } 166 } 167 168 /** 169 * Internal sorted map view. 170 * 171 * @param <K> the type of the keys. 172 * @param <V> the type of the values. 173 */ 174 protected static class ViewMap<K, V> extends AbstractSortedMapDecorator<K, V> { 175 /** 176 * Constructs a new instance. 177 * @param bidi the parent bidi map 178 * @param sm the subMap sorted map 179 */ 180 protected ViewMap(final DualTreeBidiMap<K, V> bidi, final SortedMap<K, V> sm) { 181 // the implementation is not great here... 182 // use the normalMap as the filtered map, but reverseMap as the full map 183 // this forces containsValue and clear to be overridden 184 super(new DualTreeBidiMap<>(sm, bidi.reverseMap, bidi.inverseBidiMap)); 185 } 186 187 @Override 188 public void clear() { 189 // override as default implementation uses reverseMap 190 for (final Iterator<K> it = keySet().iterator(); it.hasNext();) { 191 it.next(); 192 it.remove(); 193 } 194 } 195 196 @Override 197 public boolean containsValue(final Object value) { 198 // override as default implementation uses reverseMap 199 return decorated().normalMap.containsValue(value); 200 } 201 202 @Override 203 protected DualTreeBidiMap<K, V> decorated() { 204 return (DualTreeBidiMap<K, V>) super.decorated(); 205 } 206 207 @Override 208 public SortedMap<K, V> headMap(final K toKey) { 209 return new ViewMap<>(decorated(), super.headMap(toKey)); 210 } 211 212 @Override 213 public K nextKey(final K key) { 214 return decorated().nextKey(key); 215 } 216 217 @Override 218 public K previousKey(final K key) { 219 return decorated().previousKey(key); 220 } 221 222 @Override 223 public SortedMap<K, V> subMap(final K fromKey, final K toKey) { 224 return new ViewMap<>(decorated(), super.subMap(fromKey, toKey)); 225 } 226 227 @Override 228 public SortedMap<K, V> tailMap(final K fromKey) { 229 return new ViewMap<>(decorated(), super.tailMap(fromKey)); 230 } 231 } 232 233 /** Ensure serialization compatibility */ 234 private static final long serialVersionUID = 721969328361809L; 235 236 /** The key comparator to use */ 237 private final Comparator<? super K> comparator; 238 239 /** The value comparator to use */ 240 private final Comparator<? super V> valueComparator; 241 242 /** 243 * Creates an empty {@link DualTreeBidiMap}. 244 */ 245 public DualTreeBidiMap() { 246 super(new TreeMap<>(), new TreeMap<>()); 247 this.comparator = null; 248 this.valueComparator = null; 249 } 250 251 /** 252 * Constructs a {@link DualTreeBidiMap} using the specified {@link Comparator}. 253 * 254 * @param keyComparator the comparator 255 * @param valueComparator the values comparator to use 256 */ 257 public DualTreeBidiMap(final Comparator<? super K> keyComparator, final Comparator<? super V> valueComparator) { 258 super(new TreeMap<>(keyComparator), new TreeMap<>(valueComparator)); 259 this.comparator = keyComparator; 260 this.valueComparator = valueComparator; 261 } 262 263 /** 264 * Constructs a {@link DualTreeBidiMap} and copies the mappings from 265 * specified {@link Map}. 266 * 267 * @param map the map whose mappings are to be placed in this map 268 */ 269 public DualTreeBidiMap(final Map<? extends K, ? extends V> map) { 270 super(new TreeMap<>(), new TreeMap<>()); 271 putAll(map); 272 this.comparator = null; 273 this.valueComparator = null; 274 } 275 276 /** 277 * Constructs a {@link DualTreeBidiMap} that decorates the specified maps. 278 * 279 * @param normalMap the normal direction map 280 * @param reverseMap the reverse direction map 281 * @param inverseBidiMap the inverse BidiMap 282 */ 283 protected DualTreeBidiMap(final Map<K, V> normalMap, final Map<V, K> reverseMap, 284 final BidiMap<V, K> inverseBidiMap) { 285 super(normalMap, reverseMap, inverseBidiMap); 286 this.comparator = ((SortedMap<K, V>) normalMap).comparator(); 287 this.valueComparator = ((SortedMap<V, K>) reverseMap).comparator(); 288 } 289 290 @Override 291 public Comparator<? super K> comparator() { 292 return ((SortedMap<K, V>) normalMap).comparator(); 293 } 294 295 /** 296 * Creates a new instance of this object. 297 * 298 * @param normalMap the normal direction map 299 * @param reverseMap the reverse direction map 300 * @param inverseMap the inverse BidiMap 301 * @return new bidi map 302 */ 303 @Override 304 protected DualTreeBidiMap<V, K> createBidiMap(final Map<V, K> normalMap, final Map<K, V> reverseMap, 305 final BidiMap<K, V> inverseMap) { 306 return new DualTreeBidiMap<>(normalMap, reverseMap, inverseMap); 307 } 308 309 @Override 310 public K firstKey() { 311 return ((SortedMap<K, V>) normalMap).firstKey(); 312 } 313 314 @Override 315 public SortedMap<K, V> headMap(final K toKey) { 316 final SortedMap<K, V> sub = ((SortedMap<K, V>) normalMap).headMap(toKey); 317 return new ViewMap<>(this, sub); 318 } 319 320 @Override 321 public SortedBidiMap<V, K> inverseBidiMap() { 322 return (SortedBidiMap<V, K>) super.inverseBidiMap(); 323 } 324 325 /** 326 * Defaults to {@link #inverseBidiMap()}. 327 * 328 * @return Defaults to {@link #inverseBidiMap()}. 329 */ 330 public OrderedBidiMap<V, K> inverseOrderedBidiMap() { 331 return inverseBidiMap(); 332 } 333 334 /** 335 * Defaults to {@link #inverseBidiMap()}. 336 * 337 * @return Defaults to {@link #inverseBidiMap()}. 338 */ 339 public SortedBidiMap<V, K> inverseSortedBidiMap() { 340 return inverseBidiMap(); 341 } 342 343 @Override 344 public K lastKey() { 345 return ((SortedMap<K, V>) normalMap).lastKey(); 346 } 347 348 /** 349 * Obtains an ordered map iterator. 350 * <p> 351 * This implementation copies the elements to an ArrayList in order to 352 * provide the forward/backward behavior. 353 * </p> 354 * 355 * @return a new ordered map iterator 356 */ 357 @Override 358 public OrderedMapIterator<K, V> mapIterator() { 359 return new BidiOrderedMapIterator<>(this); 360 } 361 362 @Override 363 public K nextKey(final K key) { 364 if (isEmpty()) { 365 return null; 366 } 367 if (normalMap instanceof OrderedMap) { 368 return ((OrderedMap<K, ?>) normalMap).nextKey(key); 369 } 370 final SortedMap<K, V> sm = (SortedMap<K, V>) normalMap; 371 final Iterator<K> it = sm.tailMap(key).keySet().iterator(); 372 it.next(); 373 if (it.hasNext()) { 374 return it.next(); 375 } 376 return null; 377 } 378 379 @Override 380 public K previousKey(final K key) { 381 if (isEmpty()) { 382 return null; 383 } 384 if (normalMap instanceof OrderedMap) { 385 return ((OrderedMap<K, V>) normalMap).previousKey(key); 386 } 387 final SortedMap<K, V> sm = (SortedMap<K, V>) normalMap; 388 final SortedMap<K, V> hm = sm.headMap(key); 389 if (hm.isEmpty()) { 390 return null; 391 } 392 return hm.lastKey(); 393 } 394 395 /** 396 * Deserializes an instance from an ObjectInputStream. 397 * 398 * @param in The source ObjectInputStream. 399 * @throws IOException Any of the usual Input/Output related exceptions. 400 * @throws ClassNotFoundException A class of a serialized object cannot be found. 401 */ 402 private void readObject(final ObjectInputStream in) throws IOException, ClassNotFoundException { 403 in.defaultReadObject(); 404 normalMap = new TreeMap<>(comparator); 405 reverseMap = new TreeMap<>(valueComparator); 406 @SuppressWarnings("unchecked") // will fail at runtime if the stream is incorrect 407 final Map<K, V> map = (Map<K, V>) in.readObject(); 408 putAll(map); 409 } 410 411 @Override 412 public SortedMap<K, V> subMap(final K fromKey, final K toKey) { 413 final SortedMap<K, V> sub = ((SortedMap<K, V>) normalMap).subMap(fromKey, toKey); 414 return new ViewMap<>(this, sub); 415 } 416 417 @Override 418 public SortedMap<K, V> tailMap(final K fromKey) { 419 final SortedMap<K, V> sub = ((SortedMap<K, V>) normalMap).tailMap(fromKey); 420 return new ViewMap<>(this, sub); 421 } 422 423 @Override 424 public Comparator<? super V> valueComparator() { 425 return ((SortedMap<V, K>) reverseMap).comparator(); 426 } 427 428 /** 429 * Serializes this object to an ObjectOutputStream. 430 * 431 * @param out the target ObjectOutputStream. 432 * @throws IOException thrown when an I/O errors occur writing to the target stream. 433 */ 434 private void writeObject(final ObjectOutputStream out) throws IOException { 435 out.defaultWriteObject(); 436 out.writeObject(normalMap); 437 } 438 439}