ConcurrentHashMap/ConcurrentMap/Map.compute

Arne Siegel v.a.ammodytes at googlemail.com
Sun Dec 16 05:51:15 PST 2012


On 14 Dec 2012 at 11:03, Doug Lea wrote:

> 
> I placed a full candidate version of Map.java at
> http://gee.cs.oswego.edu/dl/wwwtmp/apis/Map.java
> 

Hi Doug,

just digged into your proposed implementation, compiling it using the lambda-8-b67 compiler, 
and observed the following issues:

1) I needed to change 
import java.util.functions.*;
into
import java.util.function.*;

2) The 'default' keyword is missing in:
    V compute(K key,
              BiFunction<? super K, ? super V, ? extends V> remappingFunction) {
...

3) Error in method signature:
    default V merge(K key, V value,
                    BiFunction<? super V, ? super V, ? extends V> remappingFunction) {
must be changed into
    default V merge(K key, V value,
                    BiFunction<? super K, ? super V, ? extends V> remappingFunction) {

4) The javadoc of 'merge' contains an odd code example, as there's no put method with 3 
arguments:

 V oldValue = map.get(key);
 V newValue = !map.containsKey(key)? value :
              remappingFunction.apply(oldValue, value);
 if (newValue != null)
   map.put(key, oldValue, newValue);
 else
   map.remove(key, oldValue);

You could write map.replace instead, but that's not the whole story, because:

5) The implementation of merge goes into an infinite loop for keys not already existing in the 
map, due to the semantics of the 'replace' method. I tested it with this code:

        Map<Integer, Integer> m = ...;
        for (int n = 0; n <= 10; n++) {
            m.put(n, n);
        }
        for (int n = 0; n <= 15; n++) {
            m.merge(n, n * n, (k, v) -> k * v);
        }

To make it work, the merge method may be corrected the following way:

replace
                if (replace(key, oldValue, newValue))
                    return newValue;
by
                if (oldValue == null ? putIfAbsent(key, newValue) == null :
                    replace(key, oldValue, newValue))
                    return newValue;


Best regards
Arne Siegel


More information about the lambda-libs-spec-observers mailing list