View Javadoc
1   /*
2    * Licensed to the Apache Software Foundation (ASF) under one or more
3    * contributor license agreements.  See the NOTICE file distributed with
4    * this work for additional information regarding copyright ownership.
5    * The ASF licenses this file to You under the Apache License, Version 2.0
6    * (the "License"); you may not use this file except in compliance with
7    * the License.  You may obtain a copy of the License at
8    *
9    *      http://www.apache.org/licenses/LICENSE-2.0
10   *
11   * Unless required by applicable law or agreed to in writing, software
12   * distributed under the License is distributed on an "AS IS" BASIS,
13   * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14   * See the License for the specific language governing permissions and
15   * limitations under the License.
16   */
17  package org.apache.commons.lang3.concurrent;
18  
19  import java.util.concurrent.ConcurrentMap;
20  import java.util.concurrent.ExecutionException;
21  import java.util.concurrent.Future;
22  import java.util.concurrent.TimeUnit;
23  
24  import org.apache.commons.lang3.Validate;
25  import org.apache.commons.lang3.exception.ExceptionUtils;
26  
27  /**
28   * An utility class providing functionality related to the {@code
29   * java.util.concurrent} package.
30   *
31   * @since 3.0
32   */
33  public class ConcurrentUtils {
34  
35      /**
36       * A specialized {@link Future} implementation which wraps a constant value.
37       * @param <T> the type of the value wrapped by this class
38       */
39      static final class ConstantFuture<T> implements Future<T> {
40          /** The constant value. */
41          private final T value;
42  
43          /**
44           * Creates a new instance of {@link ConstantFuture} and initializes it
45           * with the constant value.
46           *
47           * @param value the value (may be <b>null</b>)
48           */
49          ConstantFuture(final T value) {
50              this.value = value;
51          }
52  
53          /**
54           * {@inheritDoc} The cancel operation is not supported. This
55           * implementation always returns <b>false</b>.
56           */
57          @Override
58          public boolean cancel(final boolean mayInterruptIfRunning) {
59              return false;
60          }
61  
62          /**
63           * {@inheritDoc} This implementation just returns the constant value.
64           */
65          @Override
66          public T get() {
67              return value;
68          }
69  
70          /**
71           * {@inheritDoc} This implementation just returns the constant value; it
72           * does not block, therefore the timeout has no meaning.
73           */
74          @Override
75          public T get(final long timeout, final TimeUnit unit) {
76              return value;
77          }
78  
79          /**
80           * {@inheritDoc} This implementation always returns <b>false</b>; there
81           * is no background process which could be cancelled.
82           */
83          @Override
84          public boolean isCancelled() {
85              return false;
86          }
87  
88          /**
89           * {@inheritDoc} This implementation always returns <b>true</b> because
90           * the constant object managed by this {@link Future} implementation is
91           * always available.
92           */
93          @Override
94          public boolean isDone() {
95              return true;
96          }
97      }
98  
99      /**
100      * Tests whether the specified {@link Throwable} is a checked exception. If
101      * not, an exception is thrown.
102      *
103      * @param ex the {@link Throwable} to check
104      * @return a flag whether the passed in exception is a checked exception
105      * @throws IllegalArgumentException if the {@link Throwable} is not a
106      * checked exception
107      */
108     static Throwable checkedException(final Throwable ex) {
109         Validate.isTrue(ExceptionUtils.isChecked(ex), "Not a checked exception: " + ex);
110         return ex;
111     }
112 
113     /**
114      * Gets an implementation of {@link Future} that is immediately done
115      * and returns the specified constant value.
116      *
117      * <p>
118      * This can be useful to return a simple constant immediately from the
119      * concurrent processing, perhaps as part of avoiding nulls.
120      * A constant future can also be useful in testing.
121      * </p>
122      *
123      * @param <T> the type of the value used by this {@link Future} object
124      * @param value  the constant value to return, may be null
125      * @return an instance of Future that will return the value, never null
126      */
127     public static <T> Future<T> constantFuture(final T value) {
128         return new ConstantFuture<>(value);
129     }
130 
131     /**
132      * Checks if a concurrent map contains a key and creates a corresponding
133      * value if not. This method first checks the presence of the key in the
134      * given map. If it is already contained, its value is returned. Otherwise
135      * the {@code get()} method of the passed in {@link ConcurrentInitializer}
136      * is called. With the resulting object
137      * {@link #putIfAbsent(ConcurrentMap, Object, Object)} is called. This
138      * handles the case that in the meantime another thread has added the key to
139      * the map. Both the map and the initializer can be <b>null</b>; in this
140      * case this method simply returns <b>null</b>.
141      *
142      * @param <K> the type of the keys of the map
143      * @param <V> the type of the values of the map
144      * @param map the map to be modified
145      * @param key the key of the value to be added
146      * @param init the {@link ConcurrentInitializer} for creating the value
147      * @return the value stored in the map after this operation; this may or may
148      * not be the object created by the {@link ConcurrentInitializer}
149      * @throws ConcurrentException if the initializer throws an exception
150      */
151     public static <K, V> V createIfAbsent(final ConcurrentMap<K, V> map, final K key,
152             final ConcurrentInitializer<V> init) throws ConcurrentException {
153         if (map == null || init == null) {
154             return null;
155         }
156 
157         final V value = map.get(key);
158         if (value == null) {
159             return putIfAbsent(map, key, init.get());
160         }
161         return value;
162     }
163 
164     /**
165      * Checks if a concurrent map contains a key and creates a corresponding
166      * value if not, suppressing checked exceptions. This method calls
167      * {@code createIfAbsent()}. If a {@link ConcurrentException} is thrown, it
168      * is caught and re-thrown as a {@link ConcurrentRuntimeException}.
169      *
170      * @param <K> the type of the keys of the map
171      * @param <V> the type of the values of the map
172      * @param map the map to be modified
173      * @param key the key of the value to be added
174      * @param init the {@link ConcurrentInitializer} for creating the value
175      * @return the value stored in the map after this operation; this may or may
176      * not be the object created by the {@link ConcurrentInitializer}
177      * @throws ConcurrentRuntimeException if the initializer throws an exception
178      */
179     public static <K, V> V createIfAbsentUnchecked(final ConcurrentMap<K, V> map,
180             final K key, final ConcurrentInitializer<V> init) {
181         try {
182             return createIfAbsent(map, key, init);
183         } catch (final ConcurrentException cex) {
184             throw new ConcurrentRuntimeException(cex.getCause());
185         }
186     }
187 
188     /**
189      * Inspects the cause of the specified {@link ExecutionException} and
190      * creates a {@link ConcurrentException} with the checked cause if
191      * necessary. This method performs the following checks on the cause of the
192      * passed in exception:
193      * <ul>
194      * <li>If the passed in exception is <b>null</b> or the cause is
195      * <b>null</b>, this method returns <b>null</b>.</li>
196      * <li>If the cause is a runtime exception, it is directly thrown.</li>
197      * <li>If the cause is an error, it is directly thrown, too.</li>
198      * <li>In any other case the cause is a checked exception. The method then
199      * creates a {@link ConcurrentException}, initializes it with the cause, and
200      * returns it.</li>
201      * </ul>
202      *
203      * @param ex the exception to be processed
204      * @return a {@link ConcurrentException} with the checked cause
205      */
206     public static ConcurrentException extractCause(final ExecutionException ex) {
207         if (ex == null || ex.getCause() == null) {
208             return null;
209         }
210         ExceptionUtils.throwUnchecked(ex.getCause());
211         return new ConcurrentException(ex.getMessage(), ex.getCause());
212     }
213 
214     /**
215      * Inspects the cause of the specified {@link ExecutionException} and
216      * creates a {@link ConcurrentRuntimeException} with the checked cause if
217      * necessary. This method works exactly like
218      * {@link #extractCause(ExecutionException)}. The only difference is that
219      * the cause of the specified {@link ExecutionException} is extracted as a
220      * runtime exception. This is an alternative for client code that does not
221      * want to deal with checked exceptions.
222      *
223      * @param ex the exception to be processed
224      * @return a {@link ConcurrentRuntimeException} with the checked cause
225      */
226     public static ConcurrentRuntimeException extractCauseUnchecked(
227             final ExecutionException ex) {
228         if (ex == null || ex.getCause() == null) {
229             return null;
230         }
231 
232         ExceptionUtils.throwUnchecked(ex.getCause());
233         return new ConcurrentRuntimeException(ex.getMessage(), ex.getCause());
234     }
235 
236     /**
237      * Handles the specified {@link ExecutionException}. This method calls
238      * {@link #extractCause(ExecutionException)} for obtaining the cause of the
239      * exception - which might already cause an unchecked exception or an error
240      * being thrown. If the cause is a checked exception however, it is wrapped
241      * in a {@link ConcurrentException}, which is thrown. If the passed in
242      * exception is <b>null</b> or has no cause, the method simply returns
243      * without throwing an exception.
244      *
245      * @param ex the exception to be handled
246      * @throws ConcurrentException if the cause of the {@code
247      * ExecutionException} is a checked exception
248      */
249     public static void handleCause(final ExecutionException ex)
250             throws ConcurrentException {
251         final ConcurrentException cause = extractCause(ex);
252 
253         if (cause != null) {
254             throw cause;
255         }
256     }
257 
258     /**
259      * Handles the specified {@link ExecutionException} and transforms it into a
260      * runtime exception. This method works exactly like
261      * {@link #handleCause(ExecutionException)}, but instead of a
262      * {@link ConcurrentException} it throws a
263      * {@link ConcurrentRuntimeException}. This is an alternative for client
264      * code that does not want to deal with checked exceptions.
265      *
266      * @param ex the exception to be handled
267      * @throws ConcurrentRuntimeException if the cause of the {@code
268      * ExecutionException} is a checked exception; this exception is then
269      * wrapped in the thrown runtime exception
270      */
271     public static void handleCauseUnchecked(final ExecutionException ex) {
272         final ConcurrentRuntimeException cause = extractCauseUnchecked(ex);
273 
274         if (cause != null) {
275             throw cause;
276         }
277     }
278 
279     /**
280      * Invokes the specified {@link ConcurrentInitializer} and returns the
281      * object produced by the initializer. This method just invokes the {@code
282      * get()} method of the given {@link ConcurrentInitializer}. It is
283      * <b>null</b>-safe: if the argument is <b>null</b>, result is also
284      * <b>null</b>.
285      *
286      * @param <T> the type of the object produced by the initializer
287      * @param initializer the {@link ConcurrentInitializer} to be invoked
288      * @return the object managed by the {@link ConcurrentInitializer}
289      * @throws ConcurrentException if the {@link ConcurrentInitializer} throws
290      * an exception
291      */
292     public static <T> T initialize(final ConcurrentInitializer<T> initializer)
293             throws ConcurrentException {
294         return initializer != null ? initializer.get() : null;
295     }
296 
297     /**
298      * Invokes the specified {@link ConcurrentInitializer} and transforms
299      * occurring exceptions to runtime exceptions. This method works like
300      * {@link #initialize(ConcurrentInitializer)}, but if the {@code
301      * ConcurrentInitializer} throws a {@link ConcurrentException}, it is
302      * caught, and the cause is wrapped in a {@link ConcurrentRuntimeException}.
303      * So client code does not have to deal with checked exceptions.
304      *
305      * @param <T> the type of the object produced by the initializer
306      * @param initializer the {@link ConcurrentInitializer} to be invoked
307      * @return the object managed by the {@link ConcurrentInitializer}
308      * @throws ConcurrentRuntimeException if the initializer throws an exception
309      */
310     public static <T> T initializeUnchecked(final ConcurrentInitializer<T> initializer) {
311         try {
312             return initialize(initializer);
313         } catch (final ConcurrentException cex) {
314             throw new ConcurrentRuntimeException(cex.getCause());
315         }
316     }
317 
318     /**
319      * Puts a value in the specified {@link ConcurrentMap} if the key is not yet
320      * present. This method works similar to the {@code putIfAbsent()} method of
321      * the {@link ConcurrentMap} interface, but the value returned is different.
322      * Basically, this method is equivalent to the following code fragment:
323      *
324      * <pre>
325      * if (!map.containsKey(key)) {
326      *     map.put(key, value);
327      *     return value;
328      * } else {
329      *     return map.get(key);
330      * }
331      * </pre>
332      *
333      * <p>
334      * except that the action is performed atomically. So this method always
335      * returns the value which is stored in the map.
336      * </p>
337      * <p>
338      * This method is <b>null</b>-safe: It accepts a <b>null</b> map as input
339      * without throwing an exception. In this case the return value is
340      * <b>null</b>, too.
341      * </p>
342      *
343      * @param <K> the type of the keys of the map
344      * @param <V> the type of the values of the map
345      * @param map the map to be modified
346      * @param key the key of the value to be added
347      * @param value the value to be added
348      * @return the value stored in the map after this operation
349      */
350     public static <K, V> V putIfAbsent(final ConcurrentMap<K, V> map, final K key, final V value) {
351         if (map == null) {
352             return null;
353         }
354 
355         final V result = map.putIfAbsent(key, value);
356         return result != null ? result : value;
357     }
358 
359     /**
360      * Private constructor so that no instances can be created. This class
361      * contains only static utility methods.
362      */
363     private ConcurrentUtils() {
364     }
365 
366 }