ConcurrentHashMap/ConcurrentMap/Map.compute

Brian Goetz brian.goetz at oracle.com
Wed Dec 5 13:31:56 PST 2012


These look great to me!  This default method thing was a pretty good 
idea!  :)

Nit: uses Scala fat arrow instead of Java thin arrow in examples.

Bikeshed: The name "replaceIfPresent" seems nicer than 
"computeIfPresent" and fits in with existing replace naming.

For the value-oriented CHM methods (putIfAbsent, remove, replace), I 
think it is reasonable to move these up to Map with the obvious 
defaults, with no promises about atomicity, and adjust the ConcurrentMap 
spec to add a comment about "unlike the Map versions, these are atomic." 
  The null rules get more ad-hoc for these, but can still be reasonably 
consistent with the lambda versions; NPE on any key being null.  For 
replace(K,V,null), we can treat that as remove-if-present and 
replace(K,null) can be unconditional remove.  putIfAbsent(k, null) 
should be a no-op.  remove(key, null) becomes "remove if get() returns 
null", I guess.

ObAndIWantAPony: I would also love to have Map.merge(otherMap, 
mergeLambda) whose default uses your merge below, but for which you 
could define a better override in Map implementations that offer a 
better than element-wise merge.

On 12/5/2012 4:10 PM, Doug Lea wrote:
> On 12/05/12 07:39, Doug Lea wrote:
>
>> ... method Map.compute(). Three other similar methods
>> computeIfAbsent(), computeIfPresent(), and merge()
>> amount to special cases that are generally simpler to
>> use and more efficient when they apply.
>>
>> After a few stabs at it, I think that the only sane way
>> to spec and default-implement it is to say that, even
>> if your Map implementation otherwise allows null keys/values,
>> that this method will act as if it doesn't.
>
> Since people seem not to mind this (and Brian's and Tim's
> responses on other thread seem to even encourage it), the
> set of all four are below. Please skim through and
> see if you still agree.
>
> Brian: We had discussed also defining and default-implementing
> in Map the other ConcurrentMap methods:
>
> V putIfAbsent(K key, V value);
> boolean remove(Object key, Object value);
> boolean replace(K key, V oldValue, V newValue);
> V replace(K key, V value);
>
> Any thoughts?
>
> ....
>
>
> Additions to interface Map (with actual default implementations
> elided for now):
>
>
>      /**
>       * Attempts to compute a mapping given a key and its current
>       * mapped value (or {@code null} if there is no current
>       * mapping). The default implementation is equivalent to
>       *
>       *  <pre> {@code
>       *   V value = remappingFunction.apply(key, map.get(key));
>       *   if (value != null)
>       *     map.put(key, value);
>       *   else
>       *     map.remove(key);
>       * }</pre>
>       *
>       * If the function returns {@code null}, the mapping is removed
>       * (or remains absent if initially absent).  If the function
>       * itself throws an (unchecked) exception, the exception is
>       * rethrown to its caller, and the current mapping is left
>       * unchanged.  For example, to either create or append new
>       * messages to a value mapping:
>       *
>       * <pre> {@code
>       * Map<Key, String> map = ...;
>       * final String msg = ...;
>       * map.compute(key, (k, v, msg) => (v == null) ? msg : v + msg)</pre>
>       *
>       * <p>The default implementation makes no guarantees about
>       * synchronization or atomicity properties of this method. Any
>       * class overriding this method must specify its concurrency
>       * properties.
>       *
>       * @param key key with which the specified value is to be associated
>       * @param remappingFunction the function to compute a value
>       * @return the new value associated with the specified key, or null
> if none
>       * @throws NullPointerException if the specified key is null and
>       *         this map does not support null keys, or the
>       *         remappingFunction is null
>       * @throws RuntimeException or Error if the remappingFunction does so,
>       *         in which case the mapping is unchanged
>       */
>      V compute(K key,
>                BiFunction<? super K, ? super V, ? extends V>
> remappingFunction);
>
>      /**
>       * If the specified key is not already associated with a value,
>       * attempts to compute its value using the given mappingFunction
>       * and enters it into the map unless null.  This is equivalent to:
>       *
>       * <pre> {@code
>       * if (map.containsKey(key))
>       *   return map.get(key);
>       * value = mappingFunction.apply(key);
>       * if (value != null)
>       *   map.put(key, value);
>       * return value;}</pre>
>       *
>       * If the function returns {@code null} no mapping is recorded. If
>       * the function itself throws an (unchecked) exception, the
>       * exception is rethrown to its caller, and no mapping is
>       * recorded.  The most common usage is to construct a new object
>       * serving as an initial mapped value, or memoized result, as in:
>       *
>       *  <pre> {@code
>       * map.computeIfAbsent(key, k -> new Value(f(k)));} </pre>
>       *
>       * <p>The default implementation makes no guarantees about
>       * synchronization or atomicity properties of this method. Any
>       * class overriding this method must specify its concurrency
>       * properties.
>       *
>       * @param key key with which the specified value is to be associated
>       * @param mappingFunction the function to compute a value
>       * @return the current (existing or computed) value associated with
>       *         the specified key, or null if the computed value is null
>       * @throws NullPointerException if the specified key is null and
>       *         this map does not support null keys, or the
>       *         mappingFunction is null
>       * @throws RuntimeException or Error if the mappingFunction does so,
>       *         in which case the mapping is left unestablished
>       */
>      V computeIfAbsent(K key, Function<? super K, ? extends V>
> mappingFunction);
>
>      /**
>       * If the given key is present, attempts to compute a new mapping
>       * given the key and its current mapped value. This is equivalent
>       * to:
>       *
>       *  <pre> {@code
>       *   if (map.containsKey(key)) {
>       *     value = remappingFunction.apply(key, map.get(key));
>       *     if (value != null)
>       *       map.put(key, value);
>       *     else
>       *       map.remove(key);
>       *   }
>       * }</pre>
>       *
>       * If the function returns {@code null}, the mapping is removed
>       * (or remains absent if initially absent).  If the function
>       * itself throws an (unchecked) exception, the exception is
>       * rethrown to its caller, and the current mapping is left
>       * unchanged.
>       *
>       * <p>The default implementation makes no guarantees about
>       * synchronization or atomicity properties of this method. Any
>       * class overriding this method must specify its concurrency
>       * properties.
>       *
>       * @param key key with which the specified value is to be associated
>       * @param remappingFunction the function to compute a value
>       * @return the new value associated with the specified key, or null
> if none
>       * @throws NullPointerException if the specified key is null and
>       *         this map does not support null keys, or the
>       *         remappingFunction is null
>       * @throws RuntimeException or Error if the remappingFunction does so,
>       *         in which case the mapping is unchanged
>       */
>      computeIfPresent(K key,
>                       BiFunction<? super K, ? super V, ? extends V>
> remappingFunction);
>
>      /**
>       * If the specified key is not already associated with a value,
>       * associates it with the given value.  Otherwise, replaces the
>       * value with the results of the given remapping function, or
>       * removes if {@code null}. This is equivalent to:
>       *
>       *  <pre> {@code
>       *   V newValue;
>       *   if (!map.containsKey(key))
>       *     newValue = value;
>       *   else
>       *     newValue = remappingFunction.apply(map.get(key), value);
>       *   if (newValue != null)
>       *     map.put(key, newValue);
>       *   else
>       *     map.remove(key);
>       * }</pre>
>       *
>       * If the function returns {@code null}, the mapping is removed
>       * (or remains absent if initially absent).  If the function
>       * itself throws an (unchecked) exception, the exception is
>       * rethrown to its caller, and the current mapping is left
>       * unchanged.
>       *
>       * <p>The default implementation makes no guarantees about
>       * synchronization or atomicity properties of this method. Any
>       * class overriding this method must specify its concurrency
>       * properties.
>       *
>       * @param key key with which the specified value is to be associated
>       * @param value the value to use if absent
>       * @param remappingFunction the function to recompute a value if
> present
>       * @return the new value associated with the specified key, or null
> if none
>       * @throws NullPointerException if the specified key is null and
>       *         this map does not support null keys, or the
>       *         remappingFunction is null
>       * @throws RuntimeException or Error if the remappingFunction does so,
>       *         in which case the mapping is unchanged
>       */
>      V merge(K key, V value,
>              BiFunction<? super V, ? super V, ? extends V>
> remappingFunction);
>
>
>


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