Skip to content

Commit cd0678f

Browse files
author
Stuart Marks
committedJun 4, 2021
8199318: add idempotent copy operation for Map.Entry
Reviewed-by: alanb, psandoz, dfuchs
1 parent b27599b commit cd0678f

File tree

3 files changed

+139
-21
lines changed

3 files changed

+139
-21
lines changed
 

‎src/java.base/share/classes/java/util/AbstractMap.java

+24-7
Original file line numberDiff line numberDiff line change
@@ -592,8 +592,11 @@ private static boolean eq(Object o1, Object o2) {
592592

593593
/**
594594
* An Entry maintaining a key and a value. The value may be
595-
* changed using the {@code setValue} method. This class
596-
* facilitates the process of building custom map
595+
* changed using the {@code setValue} method. Instances of
596+
* this class are not associated with any map's entry-set view.
597+
*
598+
* @apiNote
599+
* This class facilitates the process of building custom map
597600
* implementations. For example, it may be convenient to return
598601
* arrays of {@code SimpleEntry} instances in method
599602
* {@code Map.entrySet().toArray}.
@@ -725,10 +728,21 @@ public String toString() {
725728
}
726729

727730
/**
728-
* An Entry maintaining an immutable key and value. This class
729-
* does not support method {@code setValue}. This class may be
730-
* convenient in methods that return thread-safe snapshots of
731-
* key-value mappings.
731+
* An unmodifiable Entry maintaining a key and a value. This class
732+
* does not support the {@code setValue} method. Instances of
733+
* this class are not associated with any map's entry-set view.
734+
*
735+
* @apiNote
736+
* Instances of this class are not necessarily immutable, as the key
737+
* and value may be mutable. An instance of <i>this specific class</i>
738+
* is unmodifiable, because the key and value references cannot be
739+
* changed. A reference of this <i>type</i> may not be unmodifiable,
740+
* as a subclass may be modifiable or may provide the appearance of modifiability.
741+
* <p>
742+
* This class may be convenient in methods that return thread-safe snapshots of
743+
* key-value mappings. For alternatives, see the
744+
* {@link Map#entry Map::entry} and {@link Map.Entry#copyOf Map.Entry::copyOf}
745+
* methods.
732746
*
733747
* @since 1.6
734748
*/
@@ -788,7 +802,10 @@ public V getValue() {
788802
* Replaces the value corresponding to this entry with the specified
789803
* value (optional operation). This implementation simply throws
790804
* {@code UnsupportedOperationException}, as this class implements
791-
* an <i>immutable</i> map entry.
805+
* an unmodifiable map entry.
806+
*
807+
* @implSpec
808+
* The implementation in this class always throws {@code UnsupportedOperationException}.
792809
*
793810
* @param value new value to be stored in this entry
794811
* @return (Does not return)

‎src/java.base/share/classes/java/util/Map.java

+58-8
Original file line numberDiff line numberDiff line change
@@ -393,14 +393,33 @@ public interface Map<K, V> {
393393
Set<Map.Entry<K, V>> entrySet();
394394

395395
/**
396-
* A map entry (key-value pair). The {@code Map.entrySet} method returns
397-
* a collection-view of the map, whose elements are of this class. The
398-
* <i>only</i> way to obtain a reference to a map entry is from the
399-
* iterator of this collection-view. These {@code Map.Entry} objects are
400-
* valid <i>only</i> for the duration of the iteration; more formally,
401-
* the behavior of a map entry is undefined if the backing map has been
402-
* modified after the entry was returned by the iterator, except through
403-
* the {@code setValue} operation on the map entry.
396+
* A map entry (key-value pair). The Entry may be unmodifiable, or the
397+
* value may be modifiable if the optional {@code setValue} method is
398+
* implemented. The Entry may be independent of any map, or it may represent
399+
* an entry of the entry-set view of a map.
400+
* <p>
401+
* Instances of the {@code Map.Entry} interface may be obtained by iterating
402+
* the entry-set view of a map. These instances maintain a connection to the
403+
* original, backing map. This connection to the backing map is valid
404+
* <i>only</i> for the duration of iteration over the entry-set view.
405+
* During iteration of the entry-set view, if supported by the backing map,
406+
* a change to a {@code Map.Entry}'s value via the
407+
* {@link Map.Entry#setValue setValue} method will be visible in the backing map.
408+
* The behavior of such a {@code Map.Entry} instance is undefined outside of
409+
* iteration of the map's entry-set view. It is also undefined if the backing
410+
* map has been modified after the {@code Map.Entry} was returned by the
411+
* iterator, except through the {@code Map.Entry.setValue} method. In particular,
412+
* a change to the value of a mapping in the backing map might or might not be
413+
* visible in the corresponding {@code Map.Entry} element of the entry-set view.
414+
*
415+
* @apiNote
416+
* It is possible to create a {@code Map.Entry} instance that is disconnected
417+
* from a backing map by using the {@link Map.Entry#copyOf copyOf} method. For example,
418+
* the following creates a snapshot of a map's entries that is guaranteed not to
419+
* change even if the original map is modified:
420+
* <pre> {@code
421+
* var entries = map.entrySet().stream().map(Map.Entry::copyOf).toList()
422+
* }</pre>
404423
*
405424
* @see Map#entrySet()
406425
* @since 1.2
@@ -559,6 +578,37 @@ public static <K, V> Comparator<Map.Entry<K, V>> comparingByValue(Comparator<? s
559578
return (Comparator<Map.Entry<K, V>> & Serializable)
560579
(c1, c2) -> cmp.compare(c1.getValue(), c2.getValue());
561580
}
581+
582+
/**
583+
* Returns a copy of the given {@code Map.Entry}. The returned instance is not
584+
* associated with any map. The returned instance has the same characteristics
585+
* as instances returned by the {@link Map#entry Map::entry} method.
586+
*
587+
* @apiNote
588+
* An instance obtained from a map's entry-set view has a connection to that map.
589+
* The {@code copyOf} method may be used to create a {@code Map.Entry} instance,
590+
* containing the same key and value, that is independent of any map.
591+
*
592+
* @implNote
593+
* If the given entry was obtained from a call to {@code copyOf} or {@code Map::entry},
594+
* calling {@code copyOf} will generally not create another copy.
595+
*
596+
* @param <K> the type of the key
597+
* @param <V> the type of the value
598+
* @param e the entry to be copied
599+
* @return a map entry equal to the given entry
600+
* @throws NullPointerException if e is null or if either of its key or value is null
601+
* @since 17
602+
*/
603+
@SuppressWarnings("unchecked")
604+
public static <K, V> Map.Entry<K, V> copyOf(Map.Entry<? extends K, ? extends V> e) {
605+
Objects.requireNonNull(e);
606+
if (e instanceof KeyValueHolder) {
607+
return (Map.Entry<K, V>) e;
608+
} else {
609+
return Map.entry(e.getKey(), e.getValue());
610+
}
611+
}
562612
}
563613

564614
// Comparison and hashing

‎test/jdk/java/util/Map/MapFactories.java

+57-6
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2015, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2015, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -34,7 +34,6 @@
3434
import java.util.HashMap;
3535
import java.util.List;
3636
import java.util.Map;
37-
import java.util.stream.Collectors;
3837
import java.util.stream.IntStream;
3938

4039
import org.testng.annotations.DataProvider;
@@ -45,8 +44,8 @@
4544
import static org.testng.Assert.assertNotEquals;
4645
import static org.testng.Assert.assertNotSame;
4746
import static org.testng.Assert.assertSame;
47+
import static org.testng.Assert.assertThrows;
4848
import static org.testng.Assert.assertTrue;
49-
import static org.testng.Assert.fail;
5049

5150
/*
5251
* @test
@@ -470,7 +469,7 @@ public void copyOfRejectsNullValue() {
470469
Map<Integer, String> copy = Map.copyOf(map);
471470
}
472471

473-
// Map.entry() tests
472+
// Map::entry tests
474473

475474
@Test(expectedExceptions=NullPointerException.class)
476475
public void entryWithNullKeyDisallowed() {
@@ -482,6 +481,12 @@ public void entryWithNullValueDisallowed() {
482481
Map.Entry<Integer,String> e = Map.entry(0, null);
483482
}
484483

484+
@Test
485+
public void entrySetValueDisallowed() {
486+
var e = Map.entry("a", "b");
487+
assertThrows(UnsupportedOperationException.class, () -> e.setValue("x"));
488+
}
489+
485490
@Test
486491
public void entryBasicTests() {
487492
Map.Entry<String,String> kvh1 = Map.entry("xyzzy", "plugh");
@@ -492,8 +497,54 @@ public void entryBasicTests() {
492497
assertTrue(sie.equals(kvh1));
493498
assertFalse(kvh2.equals(sie));
494499
assertFalse(sie.equals(kvh2));
495-
assertEquals(sie.hashCode(), kvh1.hashCode());
496-
assertEquals(sie.toString(), kvh1.toString());
500+
assertEquals(kvh1.hashCode(), sie.hashCode());
501+
assertEquals(kvh1.toString(), sie.toString());
502+
}
503+
504+
// Map.Entry::copyOf tests
505+
506+
@Test(expectedExceptions=NullPointerException.class)
507+
public void entryCopyNullDisallowed() {
508+
Map.Entry.copyOf(null);
509+
}
510+
511+
@Test
512+
public void entryCopyWithNullKeyDisallowed() {
513+
var e = new AbstractMap.SimpleEntry<>(null, "b");
514+
assertThrows(NullPointerException.class, () -> Map.Entry.copyOf(e));
515+
}
516+
517+
@Test
518+
public void entryCopyWithNullValueDisallowed() {
519+
var e = new AbstractMap.SimpleEntry<>("a", null);
520+
assertThrows(NullPointerException.class, () -> Map.Entry.copyOf(e));
521+
}
522+
523+
@Test
524+
public void entryCopySetValueDisallowed() {
525+
var e = new AbstractMap.SimpleEntry<>("a", "b");
526+
var c = Map.Entry.copyOf(e);
527+
assertThrows(UnsupportedOperationException.class, () -> c.setValue("x"));
528+
}
529+
530+
@Test
531+
public void entryCopyBasicTests() {
532+
Map.Entry<String,String> orig = new AbstractMap.SimpleImmutableEntry<>("xyzzy", "plugh");
533+
Map.Entry<String,String> copy1 = Map.Entry.copyOf(orig);
534+
Map.Entry<String,String> copy2 = Map.Entry.copyOf(copy1);
535+
536+
assertEquals(orig, copy1);
537+
assertEquals(copy1, orig);
538+
assertEquals(orig, copy2);
539+
assertEquals(copy2, orig);
540+
assertEquals(copy1, copy2);
541+
assertEquals(copy2, copy1);
542+
543+
assertNotSame(orig, copy1);
544+
assertSame(copy1, copy2);
545+
546+
assertEquals(copy1.hashCode(), orig.hashCode());
547+
assertEquals(copy1.toString(), orig.toString());
497548
}
498549

499550
// compile-time test of wildcards

0 commit comments

Comments
 (0)
Please sign in to comment.