| /* |
| * Copyright (C) 2010 Google Inc. |
| * |
| * 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 libcore.java.util; |
| |
| import java.util.AbstractMap.SimpleEntry; |
| import java.util.ConcurrentModificationException; |
| import java.util.HashMap; |
| import java.util.Iterator; |
| import java.util.Map; |
| import java.util.Map.Entry; |
| import java.util.NavigableMap; |
| import java.util.SortedMap; |
| import java.util.TreeMap; |
| import junit.framework.TestCase; |
| import libcore.util.SerializationTester; |
| |
| public class TreeMapTest extends TestCase { |
| |
| /** |
| * Test that the entrySet() method produces correctly mutable entries. |
| */ |
| public void testEntrySetSetValue() { |
| TreeMap<String, String> map = new TreeMap<String, String>(); |
| map.put("A", "a"); |
| map.put("B", "b"); |
| map.put("C", "c"); |
| |
| Iterator<Entry<String, String>> iterator = map.entrySet().iterator(); |
| Entry<String, String> entryA = iterator.next(); |
| assertEquals("a", entryA.setValue("x")); |
| assertEquals("x", entryA.getValue()); |
| assertEquals("x", map.get("A")); |
| Entry<String, String> entryB = iterator.next(); |
| assertEquals("b", entryB.setValue("y")); |
| Entry<String, String> entryC = iterator.next(); |
| assertEquals("c", entryC.setValue("z")); |
| assertEquals("y", entryB.getValue()); |
| assertEquals("y", map.get("B")); |
| assertEquals("z", entryC.getValue()); |
| assertEquals("z", map.get("C")); |
| } |
| |
| /** |
| * Test that the entrySet() method of a sub map produces correctly mutable |
| * entries that propagate changes to the original map. |
| */ |
| public void testSubMapEntrySetSetValue() { |
| TreeMap<String, String> map = new TreeMap<String, String>(); |
| map.put("A", "a"); |
| map.put("B", "b"); |
| map.put("C", "c"); |
| map.put("D", "d"); |
| NavigableMap<String, String> subMap = map.subMap("A", true, "C", true); |
| |
| Iterator<Entry<String, String>> iterator = subMap.entrySet().iterator(); |
| Entry<String, String> entryA = iterator.next(); |
| assertEquals("a", entryA.setValue("x")); |
| assertEquals("x", entryA.getValue()); |
| assertEquals("x", subMap.get("A")); |
| assertEquals("x", map.get("A")); |
| Entry<String, String> entryB = iterator.next(); |
| assertEquals("b", entryB.setValue("y")); |
| Entry<String, String> entryC = iterator.next(); |
| assertEquals("c", entryC.setValue("z")); |
| assertEquals("y", entryB.getValue()); |
| assertEquals("y", subMap.get("B")); |
| assertEquals("y", map.get("B")); |
| assertEquals("z", entryC.getValue()); |
| assertEquals("z", subMap.get("C")); |
| assertEquals("z", map.get("C")); |
| } |
| |
| /** |
| * Test that an Entry given by any method except entrySet() is immutable. |
| */ |
| public void testExceptionsOnSetValue() { |
| TreeMap<String, String> map = new TreeMap<String, String>(); |
| map.put("A", "a"); |
| map.put("B", "b"); |
| map.put("C", "c"); |
| |
| assertAllEntryMethodsReturnImmutableEntries(map); |
| } |
| |
| /** |
| * Test that an Entry given by any method except entrySet() of a sub map is immutable. |
| */ |
| public void testExceptionsOnSubMapSetValue() { |
| TreeMap<String, String> map = new TreeMap<String, String>(); |
| map.put("A", "a"); |
| map.put("B", "b"); |
| map.put("C", "c"); |
| map.put("D", "d"); |
| |
| assertAllEntryMethodsReturnImmutableEntries(map.subMap("A", true, "C", true)); |
| } |
| |
| /** |
| * Asserts that each NavigableMap method that returns an Entry (except entrySet()) returns an |
| * immutable one. Assumes that the map contains at least entries with keys "A", "B" and "C". |
| */ |
| private void assertAllEntryMethodsReturnImmutableEntries(NavigableMap<String, String> map) { |
| assertImmutable(map.ceilingEntry("B")); |
| assertImmutable(map.firstEntry()); |
| assertImmutable(map.floorEntry("D")); |
| assertImmutable(map.higherEntry("A")); |
| assertImmutable(map.lastEntry()); |
| assertImmutable(map.lowerEntry("C")); |
| assertImmutable(map.pollFirstEntry()); |
| assertImmutable(map.pollLastEntry()); |
| } |
| |
| private void assertImmutable(Entry<String, String> entry) { |
| String initialValue = entry.getValue(); |
| try { |
| entry.setValue("x"); |
| fail(); |
| } catch (UnsupportedOperationException e) { |
| } |
| assertEquals(initialValue, entry.getValue()); |
| } |
| |
| public void testConcurrentModificationDetection() { |
| Map<String, String> map = new TreeMap<String, String>(); |
| map.put("A", "a"); |
| map.put("B", "b"); |
| map.put("C", "c"); |
| Iterator<Entry<String,String>> iterator = map.entrySet().iterator(); |
| iterator.next(); |
| iterator.next(); |
| iterator.remove(); |
| map.put("D", "d"); |
| try { |
| iterator.next(); |
| fail(); |
| } catch (ConcurrentModificationException e) { |
| } |
| } |
| |
| public void testIteratorRemoves() { |
| Map<String, String> map = new TreeMap<String, String>(); |
| map.put("A", "a"); |
| map.put("B", "b"); |
| map.put("C", "c"); |
| Iterator<Entry<String,String>> iterator = map.entrySet().iterator(); |
| assertEquals("A", iterator.next().getKey()); |
| assertEquals("B", iterator.next().getKey()); |
| iterator.remove(); |
| assertEquals("C", iterator.next().getKey()); |
| iterator.remove(); |
| assertFalse(iterator.hasNext()); |
| } |
| |
| /** |
| * Test that entry set contains and removal use the comparator rather than equals. |
| */ |
| public void testEntrySetUsesComparatorOnly() { |
| Map<String,String> map = new TreeMap<String, String>(String.CASE_INSENSITIVE_ORDER); |
| map.put("ABC", "a"); |
| assertTrue(map.entrySet().contains(new SimpleEntry<String, String>("abc", "a"))); |
| assertTrue(map.entrySet().remove(new SimpleEntry<String, String>("abc", "a"))); |
| assertEquals(0, map.size()); |
| } |
| |
| public void testMapConstructorPassingSortedMap() { |
| Map<String,String> source = new TreeMap<String, String>(String.CASE_INSENSITIVE_ORDER); |
| NavigableMap<String,String> copy = new TreeMap<String, String>(source); |
| assertEquals(null, copy.comparator()); |
| } |
| |
| public void testNullsWithNaturalOrder() { |
| HashMap<String, String> copyFrom = new HashMap<String, String>(); |
| copyFrom.put(null, "b"); |
| try { |
| new TreeMap<String, String>(copyFrom); |
| fail(); // the RI doesn't throw if null is the only key |
| } catch (NullPointerException expected) { |
| } |
| |
| TreeMap<String,String> map = new TreeMap<String, String>(); |
| try { |
| map.put(null, "b"); |
| fail(); |
| } catch (NullPointerException e) { |
| } |
| |
| try { |
| map.descendingMap().put(null, "b"); |
| fail(); |
| } catch (NullPointerException e) { |
| } |
| |
| try { |
| map.subMap("a", "z").put(null, "b"); |
| fail(); |
| } catch (NullPointerException e) { |
| } |
| } |
| |
| public void testClassCastExceptions() { |
| Map<Object, Object> map = new TreeMap<Object, Object>(); |
| map.put("A", "a"); |
| try { |
| map.get(5); |
| fail(); |
| } catch (ClassCastException e) { |
| } |
| try { |
| map.containsKey(5); |
| fail(); |
| } catch (ClassCastException e) { |
| } |
| try { |
| map.remove(5); |
| fail(); |
| } catch (ClassCastException e) { |
| } |
| } |
| |
| public void testClassCastExceptionsOnBounds() { |
| Map<Object, Object> map = new TreeMap<Object, Object>().subMap("a", "z"); |
| try { |
| map.get(5); |
| fail(); |
| } catch (ClassCastException e) { |
| } |
| try { |
| map.containsKey(5); |
| fail(); |
| } catch (ClassCastException e) { |
| } |
| try { |
| map.remove(5); |
| fail(); |
| } catch (ClassCastException e) { |
| } |
| } |
| |
| public void testClone() { |
| TreeMap<String, String> map = new TreeMap<String, String>() { |
| @Override public String toString() { |
| return "x:" + super.toString(); |
| } |
| }; |
| |
| map.put("A", "a"); |
| map.put("B", "b"); |
| |
| @SuppressWarnings("unchecked") |
| TreeMap<String, String> clone = (TreeMap<String, String>) map.clone(); |
| assertEquals(map.getClass(), clone.getClass()); |
| assertEquals("x:{A=a, B=b}", map.toString()); |
| } |
| |
| public void testEmptyMapSerialization() { |
| String s = "aced0005737200116a6176612e7574696c2e547265654d61700cc1f63e2d256a" |
| + "e60300014c000a636f6d70617261746f727400164c6a6176612f7574696c2f436" |
| + "f6d70617261746f723b78707077040000000078"; |
| TreeMap<String, String> map = new TreeMap<String, String>(); |
| new SerializationTester<TreeMap<String, String>>(map, s).test(); |
| } |
| |
| public void testSerializationWithComparator() { |
| String s = "aced0005737200116a6176612e7574696c2e547265654d61700cc1f63e2d256a" |
| + "e60300014c000a636f6d70617261746f727400164c6a6176612f7574696c2f436" |
| + "f6d70617261746f723b78707372002a6a6176612e6c616e672e537472696e6724" |
| + "43617365496e73656e736974697665436f6d70617261746f7277035c7d5c50e5c" |
| + "e02000078707704000000027400016171007e00057400016271007e000678"; |
| TreeMap<String, String> map = new TreeMap<String, String>( |
| String.CASE_INSENSITIVE_ORDER); |
| map.put("a", "a"); |
| map.put("b", "b"); |
| new SerializationTester<NavigableMap<String, String>>(map, s) { |
| @Override protected void verify(NavigableMap<String, String> deserialized) { |
| assertEquals(0, deserialized.comparator().compare("X", "x")); |
| } |
| }.test(); |
| } |
| |
| public void testSubMapSerialization() { |
| String s = "aced0005737200216a6176612e7574696c2e547265654d617024417363656e646" |
| + "96e675375624d61700cab946d1f0fab1c020000787200216a6176612e7574696c2" |
| + "e547265654d6170244e6176696761626c655375624d6170e2d0a70e64210e08020" |
| + "0075a000966726f6d53746172745a000b6869496e636c75736976655a000b6c6f4" |
| + "96e636c75736976655a0005746f456e644c000268697400124c6a6176612f6c616" |
| + "e672f4f626a6563743b4c00026c6f71007e00024c00016d7400134c6a6176612f7" |
| + "574696c2f547265654d61703b7870000001007400016374000161737200116a617" |
| + "6612e7574696c2e547265654d61700cc1f63e2d256ae60300014c000a636f6d706" |
| + "17261746f727400164c6a6176612f7574696c2f436f6d70617261746f723b78707" |
| + "372002a6a6176612e6c616e672e537472696e672443617365496e73656e7369746" |
| + "97665436f6d70617261746f7277035c7d5c50e5ce0200007870770400000004710" |
| + "07e000671007e00067400016271007e000c71007e000571007e000574000164710" |
| + "07e000d78"; |
| TreeMap<String, String> map = new TreeMap<String, String>( |
| String.CASE_INSENSITIVE_ORDER); |
| map.put("a", "a"); |
| map.put("b", "b"); |
| map.put("c", "c"); |
| map.put("d", "d"); |
| SortedMap<String, String> subMap = map.subMap("a", "c"); |
| new SerializationTester<SortedMap<String, String>>(subMap, s) { |
| @Override protected void verify(SortedMap<String, String> deserialized) { |
| try { |
| deserialized.put("e", "e"); |
| fail(); |
| } catch (IllegalArgumentException expected) { |
| } |
| } |
| }.test(); |
| } |
| |
| public void testNavigableSubMapSerialization() { |
| String s = "aced0005737200216a6176612e7574696c2e547265654d617024417363656e646" |
| + "96e675375624d61700cab946d1f0fab1c020000787200216a6176612e7574696c2" |
| + "e547265654d6170244e6176696761626c655375624d6170e2d0a70e64210e08020" |
| + "0075a000966726f6d53746172745a000b6869496e636c75736976655a000b6c6f4" |
| + "96e636c75736976655a0005746f456e644c000268697400124c6a6176612f6c616" |
| + "e672f4f626a6563743b4c00026c6f71007e00024c00016d7400134c6a6176612f7" |
| + "574696c2f547265654d61703b7870000100007400016374000161737200116a617" |
| + "6612e7574696c2e547265654d61700cc1f63e2d256ae60300014c000a636f6d706" |
| + "17261746f727400164c6a6176612f7574696c2f436f6d70617261746f723b78707" |
| + "372002a6a6176612e6c616e672e537472696e672443617365496e73656e7369746" |
| + "97665436f6d70617261746f7277035c7d5c50e5ce0200007870770400000004710" |
| + "07e000671007e00067400016271007e000c71007e000571007e000574000164710" |
| + "07e000d78"; |
| TreeMap<String, String> map = new TreeMap<String, String>( |
| String.CASE_INSENSITIVE_ORDER); |
| map.put("a", "a"); |
| map.put("b", "b"); |
| map.put("c", "c"); |
| map.put("d", "d"); |
| SortedMap<String, String> subMap = map.subMap("a", false, "c", true); |
| new SerializationTester<SortedMap<String, String>>(subMap, s) { |
| @Override protected void verify(SortedMap<String, String> deserialized) { |
| try { |
| deserialized.put("e", "e"); |
| fail(); |
| } catch (IllegalArgumentException expected) { |
| } |
| } |
| }.test(); |
| } |
| |
| public void testDescendingMapSerialization() { |
| String s = "aced0005737200226a6176612e7574696c2e547265654d61702444657363656e6" |
| + "4696e675375624d61700cab946d1f0f9d0c0200014c001172657665727365436f6" |
| + "d70617261746f727400164c6a6176612f7574696c2f436f6d70617261746f723b7" |
| + "87200216a6176612e7574696c2e547265654d6170244e6176696761626c6553756" |
| + "24d6170e2d0a70e64210e080200075a000966726f6d53746172745a000b6869496" |
| + "e636c75736976655a000b6c6f496e636c75736976655a0005746f456e644c00026" |
| + "8697400124c6a6176612f6c616e672f4f626a6563743b4c00026c6f71007e00034" |
| + "c00016d7400134c6a6176612f7574696c2f547265654d61703b787001010101707" |
| + "0737200116a6176612e7574696c2e547265654d61700cc1f63e2d256ae60300014" |
| + "c000a636f6d70617261746f7271007e000178707372002a6a6176612e6c616e672" |
| + "e537472696e672443617365496e73656e736974697665436f6d70617261746f727" |
| + "7035c7d5c50e5ce02000078707704000000027400016171007e000a74000162710" |
| + "07e000b78737200286a6176612e7574696c2e436f6c6c656374696f6e732452657" |
| + "665727365436f6d70617261746f7232000003fa6c354d510200014c0003636d707" |
| + "1007e0001787071007e0009"; |
| TreeMap<String, String> map = new TreeMap<String, String>( |
| String.CASE_INSENSITIVE_ORDER); |
| map.put("a", "a"); |
| map.put("b", "b"); |
| NavigableMap<String, String> descendingMap = map.descendingMap(); |
| new SerializationTester<NavigableMap<String, String>>(descendingMap, s) { |
| @Override protected void verify(NavigableMap<String, String> deserialized) { |
| assertEquals("b", deserialized.navigableKeySet().first()); |
| } |
| }.test(); |
| } |
| |
| public void testJava5SerializationWithComparator() { |
| String s = "aced0005737200116a6176612e7574696c2e547265654d61700cc1f63e2d256a" |
| + "e60300014c000a636f6d70617261746f727400164c6a6176612f7574696c2f436" |
| + "f6d70617261746f723b78707372002a6a6176612e6c616e672e537472696e6724" |
| + "43617365496e73656e736974697665436f6d70617261746f7277035c7d5c50e5c" |
| + "e02000078707704000000027400016171007e00057400016271007e000678"; |
| TreeMap<String, String> map = new TreeMap<String, String>( |
| String.CASE_INSENSITIVE_ORDER); |
| map.put("a", "a"); |
| map.put("b", "b"); |
| new SerializationTester<TreeMap<String, String>>(map, s) { |
| @Override protected void verify(TreeMap<String, String> deserialized) { |
| assertEquals(0, deserialized.comparator().compare("X", "x")); |
| } |
| }.test(); |
| } |
| |
| /** |
| * On JDK5, this fails with a NullPointerException after deserialization! |
| */ |
| public void testJava5SubMapSerialization() { |
| String s = "aced0005737200186a6176612e7574696c2e547265654d6170245375624d6170" |
| + "a5818343a213c27f0200055a000966726f6d53746172745a0005746f456e644c0" |
| + "00766726f6d4b65797400124c6a6176612f6c616e672f4f626a6563743b4c0006" |
| + "7468697324307400134c6a6176612f7574696c2f547265654d61703b4c0005746" |
| + "f4b657971007e00017870000074000161737200116a6176612e7574696c2e5472" |
| + "65654d61700cc1f63e2d256ae60300014c000a636f6d70617261746f727400164" |
| + "c6a6176612f7574696c2f436f6d70617261746f723b78707372002a6a6176612e" |
| + "6c616e672e537472696e672443617365496e73656e736974697665436f6d70617" |
| + "261746f7277035c7d5c50e5ce020000787077040000000471007e000471007e00" |
| + "047400016271007e000a7400016371007e000b7400016471007e000c7871007e0" |
| + "00b"; |
| TreeMap<String, String> map = new TreeMap<String, String>(String.CASE_INSENSITIVE_ORDER); |
| map.put("a", "a"); |
| map.put("b", "b"); |
| map.put("c", "c"); |
| map.put("d", "d"); |
| SortedMap<String, String> subMap = map.subMap("a", "c"); |
| new SerializationTester<SortedMap<String, String>>(subMap, s) { |
| @Override protected void verify(SortedMap<String, String> deserialized) { |
| try { |
| deserialized.put("e", "e"); |
| fail(); |
| } catch (IllegalArgumentException expected) { |
| } |
| } |
| }.test(); |
| } |
| } |