Functions.java

  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;

  18. import java.io.IOException;
  19. import java.io.UncheckedIOException;
  20. import java.lang.reflect.UndeclaredThrowableException;
  21. import java.util.Arrays;
  22. import java.util.Collection;
  23. import java.util.Objects;
  24. import java.util.concurrent.Callable;
  25. import java.util.function.BiConsumer;
  26. import java.util.function.BiFunction;
  27. import java.util.function.BiPredicate;
  28. import java.util.function.Consumer;
  29. import java.util.function.Function;
  30. import java.util.function.Predicate;
  31. import java.util.function.Supplier;
  32. import java.util.stream.Stream;

  33. import org.apache.commons.lang3.Streams.FailableStream;
  34. import org.apache.commons.lang3.exception.ExceptionUtils;
  35. import org.apache.commons.lang3.function.Failable;
  36. import org.apache.commons.lang3.function.FailableBooleanSupplier;

  37. /**
  38.  * This class provides utility functions, and classes for working with the {@code java.util.function} package, or more
  39.  * generally, with Java 8 lambdas. More specifically, it attempts to address the fact that lambdas are supposed not to
  40.  * throw Exceptions, at least not checked Exceptions, AKA instances of {@link Exception}. This enforces the use of
  41.  * constructs like:
  42.  *
  43.  * <pre>
  44.  * {@code
  45.  *     Consumer<java.lang.reflect.Method> consumer = m -> {
  46.  *         try {
  47.  *             m.invoke(o, args);
  48.  *         } catch (Throwable t) {
  49.  *             throw Functions.rethrow(t);
  50.  *         }
  51.  *     };
  52.  * }</pre>
  53.  *
  54.  * <p>
  55.  * By replacing a {@link java.util.function.Consumer Consumer&lt;O&gt;} with a {@link FailableConsumer
  56.  * FailableConsumer&lt;O,? extends Throwable&gt;}, this can be written like follows:
  57.  * </p>
  58.  *
  59.  * <pre>
  60.  * {@code
  61.  *   Functions.accept((m) -> m.invoke(o,args));
  62.  * }</pre>
  63.  *
  64.  * <p>
  65.  * Obviously, the second version is much more concise and the spirit of Lambda expressions is met better than the second
  66.  * version.
  67.  * </p>
  68.  * @since 3.9
  69.  * @deprecated Use {@link org.apache.commons.lang3.function.Failable}.
  70.  */
  71. @Deprecated
  72. public class Functions {

  73.     /**
  74.      * A functional interface like {@link BiConsumer} that declares a {@link Throwable}.
  75.      *
  76.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  77.      *
  78.      * @param <O1> Consumed type 1.
  79.      * @param <O2> Consumed type 2.
  80.      * @param <T> Thrown exception.
  81.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableBiConsumer}.
  82.      */
  83.     @Deprecated
  84.     @FunctionalInterface
  85.     public interface FailableBiConsumer<O1, O2, T extends Throwable> {

  86.         /**
  87.          * Accepts the consumer.
  88.          *
  89.          * @param object1 the first parameter for the consumable to accept
  90.          * @param object2 the second parameter for the consumable to accept
  91.          * @throws T Thrown when the consumer fails.
  92.          */
  93.         void accept(O1 object1, O2 object2) throws T;
  94.     }

  95.     /**
  96.      * A functional interface like {@link BiFunction} that declares a {@link Throwable}.
  97.      *
  98.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  99.      *
  100.      * @param <O1> Input type 1.
  101.      * @param <O2> Input type 2.
  102.      * @param <R> Return type.
  103.      * @param <T> Thrown exception.
  104.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableBiFunction}.
  105.      */
  106.     @Deprecated
  107.     @FunctionalInterface
  108.     public interface FailableBiFunction<O1, O2, R, T extends Throwable> {

  109.         /**
  110.          * Applies this function.
  111.          *
  112.          * @param input1 the first input for the function
  113.          * @param input2 the second input for the function
  114.          * @return the result of the function
  115.          * @throws T Thrown when the function fails.
  116.          */
  117.         R apply(O1 input1, O2 input2) throws T;
  118.     }

  119.     /**
  120.      * A functional interface like {@link BiPredicate} that declares a {@link Throwable}.
  121.      *
  122.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  123.      *
  124.      * @param <O1> Predicate type 1.
  125.      * @param <O2> Predicate type 2.
  126.      * @param <T> Thrown exception.
  127.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableBiPredicate}.
  128.      */
  129.     @Deprecated
  130.     @FunctionalInterface
  131.     public interface FailableBiPredicate<O1, O2, T extends Throwable> {

  132.         /**
  133.          * Tests the predicate.
  134.          *
  135.          * @param object1 the first object to test the predicate on
  136.          * @param object2 the second object to test the predicate on
  137.          * @return the predicate's evaluation
  138.          * @throws T if the predicate fails
  139.          */
  140.         boolean test(O1 object1, O2 object2) throws T;
  141.     }

  142.     /**
  143.      * A functional interface like {@link java.util.concurrent.Callable} that declares a {@link Throwable}.
  144.      *
  145.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  146.      *
  147.      * @param <R> Return type.
  148.      * @param <T> Thrown exception.
  149.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableCallable}.
  150.      */
  151.     @Deprecated
  152.     @FunctionalInterface
  153.     public interface FailableCallable<R, T extends Throwable> {

  154.         /**
  155.          * Calls the callable.
  156.          *
  157.          * @return The value returned from the callable
  158.          * @throws T if the callable fails
  159.          */
  160.         R call() throws T;
  161.     }

  162.     /**
  163.      * A functional interface like {@link Consumer} that declares a {@link Throwable}.
  164.      *
  165.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  166.      *
  167.      * @param <O> Consumed type 1.
  168.      * @param <T> Thrown exception.
  169.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableConsumer}.
  170.      */
  171.     @Deprecated
  172.     @FunctionalInterface
  173.     public interface FailableConsumer<O, T extends Throwable> {

  174.         /**
  175.          * Accepts the consumer.
  176.          *
  177.          * @param object the parameter for the consumable to accept
  178.          * @throws T Thrown when the consumer fails.
  179.          */
  180.         void accept(O object) throws T;
  181.     }

  182.     /**
  183.      * A functional interface like {@link Function} that declares a {@link Throwable}.
  184.      *
  185.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  186.      *
  187.      * @param <I> Input type 1.
  188.      * @param <R> Return type.
  189.      * @param <T> Thrown exception.
  190.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableFunction}.
  191.      */
  192.     @Deprecated
  193.     @FunctionalInterface
  194.     public interface FailableFunction<I, R, T extends Throwable> {

  195.         /**
  196.          * Applies this function.
  197.          *
  198.          * @param input the input for the function
  199.          * @return the result of the function
  200.          * @throws T Thrown when the function fails.
  201.          */
  202.         R apply(I input) throws T;
  203.     }

  204.     /**
  205.      * A functional interface like {@link Predicate} that declares a {@link Throwable}.
  206.      *
  207.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  208.      *
  209.      * @param <I> Predicate type 1.
  210.      * @param <T> Thrown exception.
  211.      * @deprecated Use {@link org.apache.commons.lang3.function.FailablePredicate}.
  212.      */
  213.     @Deprecated
  214.     @FunctionalInterface
  215.     public interface FailablePredicate<I, T extends Throwable> {

  216.         /**
  217.          * Tests the predicate.
  218.          *
  219.          * @param object the object to test the predicate on
  220.          * @return the predicate's evaluation
  221.          * @throws T if the predicate fails
  222.          */
  223.         boolean test(I object) throws T;
  224.     }

  225.     /**
  226.      * A functional interface like {@link Runnable} that declares a {@link Throwable}.
  227.      *
  228.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  229.      *
  230.      * @param <T> Thrown exception.
  231.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableRunnable}.
  232.      */
  233.     @Deprecated
  234.     @FunctionalInterface
  235.     public interface FailableRunnable<T extends Throwable> {

  236.         /**
  237.          * Runs the function.
  238.          *
  239.          * @throws T Thrown when the function fails.
  240.          */
  241.         void run() throws T;
  242.     }

  243.     /**
  244.      * A functional interface like {@link Supplier} that declares a {@link Throwable}.
  245.      *
  246.      * <p>TODO for 4.0: Move to org.apache.commons.lang3.function.</p>
  247.      *
  248.      * @param <R> Return type.
  249.      * @param <T> Thrown exception.
  250.      * @deprecated Use {@link org.apache.commons.lang3.function.FailableSupplier}.
  251.      */
  252.     @Deprecated
  253.     @FunctionalInterface
  254.     public interface FailableSupplier<R, T extends Throwable> {

  255.         /**
  256.          * Supplies an object
  257.          *
  258.          * @return a result
  259.          * @throws T if the supplier fails
  260.          */
  261.         R get() throws T;
  262.     }

  263.     /**
  264.      * Consumes a consumer and rethrows any exception as a {@link RuntimeException}.
  265.      *
  266.      * @param consumer the consumer to consume
  267.      * @param object1 the first object to consume by {@code consumer}
  268.      * @param object2 the second object to consume by {@code consumer}
  269.      * @param <O1> the type of the first argument the consumer accepts
  270.      * @param <O2> the type of the second argument the consumer accepts
  271.      * @param <T> the type of checked exception the consumer may throw
  272.      */
  273.     public static <O1, O2, T extends Throwable> void accept(final FailableBiConsumer<O1, O2, T> consumer,
  274.         final O1 object1, final O2 object2) {
  275.         run(() -> consumer.accept(object1, object2));
  276.     }

  277.     /**
  278.      * Consumes a consumer and rethrows any exception as a {@link RuntimeException}.
  279.      *
  280.      * @param consumer the consumer to consume
  281.      * @param object the object to consume by {@code consumer}
  282.      * @param <O> the type the consumer accepts
  283.      * @param <T> the type of checked exception the consumer may throw
  284.      */
  285.     public static <O, T extends Throwable> void accept(final FailableConsumer<O, T> consumer, final O object) {
  286.         run(() -> consumer.accept(object));
  287.     }

  288.     /**
  289.      * Applies a function and rethrows any exception as a {@link RuntimeException}.
  290.      *
  291.      * @param function the function to apply
  292.      * @param input1 the first input to apply {@code function} on
  293.      * @param input2 the second input to apply {@code function} on
  294.      * @param <O1> the type of the first argument the function accepts
  295.      * @param <O2> the type of the second argument the function accepts
  296.      * @param <O> the return type of the function
  297.      * @param <T> the type of checked exception the function may throw
  298.      * @return the value returned from the function
  299.      */
  300.     public static <O1, O2, O, T extends Throwable> O apply(final FailableBiFunction<O1, O2, O, T> function,
  301.         final O1 input1, final O2 input2) {
  302.         return get(() -> function.apply(input1, input2));
  303.     }

  304.     /**
  305.      * Applies a function and rethrows any exception as a {@link RuntimeException}.
  306.      *
  307.      * @param function the function to apply
  308.      * @param input the input to apply {@code function} on
  309.      * @param <I> the type of the argument the function accepts
  310.      * @param <O> the return type of the function
  311.      * @param <T> the type of checked exception the function may throw
  312.      * @return the value returned from the function
  313.      */
  314.     public static <I, O, T extends Throwable> O apply(final FailableFunction<I, O, T> function, final I input) {
  315.         return get(() -> function.apply(input));
  316.     }

  317.     /**
  318.      * Converts the given {@link FailableBiConsumer} into a standard {@link BiConsumer}.
  319.      *
  320.      * @param <O1> the type of the first argument of the consumers
  321.      * @param <O2> the type of the second argument of the consumers
  322.      * @param consumer a failable {@link BiConsumer}
  323.      * @return a standard {@link BiConsumer}
  324.      * @since 3.10
  325.      */
  326.     public static <O1, O2> BiConsumer<O1, O2> asBiConsumer(final FailableBiConsumer<O1, O2, ?> consumer) {
  327.         return (input1, input2) -> accept(consumer, input1, input2);
  328.     }

  329.     /**
  330.      * Converts the given {@link FailableBiFunction} into a standard {@link BiFunction}.
  331.      *
  332.      * @param <O1> the type of the first argument of the input of the functions
  333.      * @param <O2> the type of the second argument of the input of the functions
  334.      * @param <O> the type of the output of the functions
  335.      * @param function a {@link FailableBiFunction}
  336.      * @return a standard {@link BiFunction}
  337.      * @since 3.10
  338.      */
  339.     public static <O1, O2, O> BiFunction<O1, O2, O> asBiFunction(final FailableBiFunction<O1, O2, O, ?> function) {
  340.         return (input1, input2) -> apply(function, input1, input2);
  341.     }

  342.     /**
  343.      * Converts the given {@link FailableBiPredicate} into a standard {@link BiPredicate}.
  344.      *
  345.      * @param <O1> the type of the first argument used by the predicates
  346.      * @param <O2> the type of the second argument used by the predicates
  347.      * @param predicate a {@link FailableBiPredicate}
  348.      * @return a standard {@link BiPredicate}
  349.      * @since 3.10
  350.      */
  351.     public static <O1, O2> BiPredicate<O1, O2> asBiPredicate(final FailableBiPredicate<O1, O2, ?> predicate) {
  352.         return (input1, input2) -> test(predicate, input1, input2);
  353.     }

  354.     /**
  355.      * Converts the given {@link FailableCallable} into a standard {@link Callable}.
  356.      *
  357.      * @param <O> the type used by the callables
  358.      * @param callable a {@link FailableCallable}
  359.      * @return a standard {@link Callable}
  360.      * @since 3.10
  361.      */
  362.     public static <O> Callable<O> asCallable(final FailableCallable<O, ?> callable) {
  363.         return () -> call(callable);
  364.     }

  365.     /**
  366.      * Converts the given {@link FailableConsumer} into a standard {@link Consumer}.
  367.      *
  368.      * @param <I> the type used by the consumers
  369.      * @param consumer a {@link FailableConsumer}
  370.      * @return a standard {@link Consumer}
  371.      * @since 3.10
  372.      */
  373.     public static <I> Consumer<I> asConsumer(final FailableConsumer<I, ?> consumer) {
  374.         return input -> accept(consumer, input);
  375.     }

  376.     /**
  377.      * Converts the given {@link FailableFunction} into a standard {@link Function}.
  378.      *
  379.      * @param <I> the type of the input of the functions
  380.      * @param <O> the type of the output of the functions
  381.      * @param function a {code FailableFunction}
  382.      * @return a standard {@link Function}
  383.      * @since 3.10
  384.      */
  385.     public static <I, O> Function<I, O> asFunction(final FailableFunction<I, O, ?> function) {
  386.         return input -> apply(function, input);
  387.     }

  388.     /**
  389.      * Converts the given {@link FailablePredicate} into a standard {@link Predicate}.
  390.      *
  391.      * @param <I> the type used by the predicates
  392.      * @param predicate a {@link FailablePredicate}
  393.      * @return a standard {@link Predicate}
  394.      * @since 3.10
  395.      */
  396.     public static <I> Predicate<I> asPredicate(final FailablePredicate<I, ?> predicate) {
  397.         return input -> test(predicate, input);
  398.     }

  399.     /**
  400.      * Converts the given {@link FailableRunnable} into a standard {@link Runnable}.
  401.      *
  402.      * @param runnable a {@link FailableRunnable}
  403.      * @return a standard {@link Runnable}
  404.      * @since 3.10
  405.      */
  406.     public static Runnable asRunnable(final FailableRunnable<?> runnable) {
  407.         return () -> run(runnable);
  408.     }

  409.     /**
  410.      * Converts the given {@link FailableSupplier} into a standard {@link Supplier}.
  411.      *
  412.      * @param <O> the type supplied by the suppliers
  413.      * @param supplier a {@link FailableSupplier}
  414.      * @return a standard {@link Supplier}
  415.      * @since 3.10
  416.      */
  417.     public static <O> Supplier<O> asSupplier(final FailableSupplier<O, ?> supplier) {
  418.         return () -> get(supplier);
  419.     }

  420.     /**
  421.      * Calls a callable and rethrows any exception as a {@link RuntimeException}.
  422.      *
  423.      * @param callable the callable to call
  424.      * @param <O> the return type of the callable
  425.      * @param <T> the type of checked exception the callable may throw
  426.      * @return the value returned from the callable
  427.      */
  428.     public static <O, T extends Throwable> O call(final FailableCallable<O, T> callable) {
  429.         return get(callable::call);
  430.     }

  431.     /**
  432.      * Invokes a supplier, and returns the result.
  433.      *
  434.      * @param supplier The supplier to invoke.
  435.      * @param <O> The suppliers output type.
  436.      * @param <T> The type of checked exception, which the supplier can throw.
  437.      * @return The object, which has been created by the supplier
  438.      * @since 3.10
  439.      */
  440.     public static <O, T extends Throwable> O get(final FailableSupplier<O, T> supplier) {
  441.         try {
  442.             return supplier.get();
  443.         } catch (final Throwable t) {
  444.             throw rethrow(t);
  445.         }
  446.     }

  447.     /**
  448.      * Invokes a boolean supplier, and returns the result.
  449.      *
  450.      * @param supplier The boolean supplier to invoke.
  451.      * @param <T> The type of checked exception, which the supplier can throw.
  452.      * @return The boolean, which has been created by the supplier
  453.      */
  454.     private static <T extends Throwable> boolean getAsBoolean(final FailableBooleanSupplier<T> supplier) {
  455.         try {
  456.             return supplier.getAsBoolean();
  457.         } catch (final Throwable t) {
  458.             throw rethrow(t);
  459.         }
  460.     }

  461.     /**
  462.      * Rethrows a {@link Throwable} as an unchecked exception. If the argument is already unchecked, namely a
  463.      * {@link RuntimeException} or {@link Error} then the argument will be rethrown without modification. If the
  464.      * exception is {@link IOException} then it will be wrapped into a {@link UncheckedIOException}. In every other
  465.      * cases the exception will be wrapped into a {@code
  466.      * UndeclaredThrowableException}
  467.      *
  468.      * <p>
  469.      * Note that there is a declared return type for this method, even though it never returns. The reason for that is
  470.      * to support the usual pattern:
  471.      * </p>
  472.      *
  473.      * <pre>
  474.      * throw rethrow(myUncheckedException);</pre>
  475.      *
  476.      * <p>
  477.      * instead of just calling the method. This pattern may help the Java compiler to recognize that at that point an
  478.      * exception will be thrown and the code flow analysis will not demand otherwise mandatory commands that could
  479.      * follow the method call, like a {@code return} statement from a value returning method.
  480.      * </p>
  481.      *
  482.      * @param throwable The throwable to rethrow possibly wrapped into an unchecked exception
  483.      * @return Never returns anything, this method never terminates normally.
  484.      */
  485.     public static RuntimeException rethrow(final Throwable throwable) {
  486.         Objects.requireNonNull(throwable, "throwable");
  487.         ExceptionUtils.throwUnchecked(throwable);
  488.         if (throwable instanceof IOException) {
  489.             throw new UncheckedIOException((IOException) throwable);
  490.         }
  491.         throw new UndeclaredThrowableException(throwable);
  492.     }

  493.     /**
  494.      * Runs a runnable and rethrows any exception as a {@link RuntimeException}.
  495.      *
  496.      * @param runnable The runnable to run
  497.      * @param <T> the type of checked exception the runnable may throw
  498.      */
  499.     public static <T extends Throwable> void run(final FailableRunnable<T> runnable) {
  500.         try {
  501.             runnable.run();
  502.         } catch (final Throwable t) {
  503.             throw rethrow(t);
  504.         }
  505.     }

  506.     /**
  507.      * Converts the given collection into a {@link FailableStream}. The {@link FailableStream} consists of the
  508.      * collections elements. Shortcut for
  509.      *
  510.      * <pre>
  511.      * Functions.stream(collection.stream());</pre>
  512.      *
  513.      * @param collection The collection, which is being converted into a {@link FailableStream}.
  514.      * @param <O> The collections element type. (In turn, the result streams element type.)
  515.      * @return The created {@link FailableStream}.
  516.      * @since 3.10
  517.      */
  518.     public static <O> FailableStream<O> stream(final Collection<O> collection) {
  519.         return new FailableStream<>(collection.stream());
  520.     }

  521.     /**
  522.      * Converts the given stream into a {@link FailableStream}. The {@link FailableStream} consists of the same
  523.      * elements, than the input stream. However, failable lambdas, like {@link FailablePredicate},
  524.      * {@link FailableFunction}, and {@link FailableConsumer} may be applied, rather than {@link Predicate},
  525.      * {@link Function}, {@link Consumer}, etc.
  526.      *
  527.      * @param stream The stream, which is being converted into a {@link FailableStream}.
  528.      * @param <O> The streams element type.
  529.      * @return The created {@link FailableStream}.
  530.      * @since 3.10
  531.      */
  532.     public static <O> FailableStream<O> stream(final Stream<O> stream) {
  533.         return new FailableStream<>(stream);
  534.     }

  535.     /**
  536.      * Tests a predicate and rethrows any exception as a {@link RuntimeException}.
  537.      *
  538.      * @param predicate the predicate to test
  539.      * @param object1 the first input to test by {@code predicate}
  540.      * @param object2 the second input to test by {@code predicate}
  541.      * @param <O1> the type of the first argument the predicate tests
  542.      * @param <O2> the type of the second argument the predicate tests
  543.      * @param <T> the type of checked exception the predicate may throw
  544.      * @return the boolean value returned by the predicate
  545.      */
  546.     public static <O1, O2, T extends Throwable> boolean test(final FailableBiPredicate<O1, O2, T> predicate,
  547.         final O1 object1, final O2 object2) {
  548.         return getAsBoolean(() -> predicate.test(object1, object2));
  549.     }

  550.     /**
  551.      * Tests a predicate and rethrows any exception as a {@link RuntimeException}.
  552.      *
  553.      * @param predicate the predicate to test
  554.      * @param object the input to test by {@code predicate}
  555.      * @param <O> the type of argument the predicate tests
  556.      * @param <T> the type of checked exception the predicate may throw
  557.      * @return the boolean value returned by the predicate
  558.      */
  559.     public static <O, T extends Throwable> boolean test(final FailablePredicate<O, T> predicate, final O object) {
  560.         return getAsBoolean(() -> predicate.test(object));
  561.     }

  562.     /**
  563.      * A simple try-with-resources implementation, that can be used, if your objects do not implement the
  564.      * {@link AutoCloseable} interface. The method executes the {@code action}. The method guarantees, that <em>all</em>
  565.      * the {@code resources} are being executed, in the given order, afterwards, and regardless of success, or failure.
  566.      * If either the original action, or any of the resource action fails, then the <em>first</em> failure (AKA
  567.      * {@link Throwable}) is rethrown. Example use:
  568.      *
  569.      * <pre>
  570.      * {@code
  571.      *     final FileInputStream fis = new FileInputStream("my.file");
  572.      *     Functions.tryWithResources(useInputStream(fis), null, () -> fis.close());
  573.      * }</pre>
  574.      *
  575.      * @param action The action to execute. This object <em>will</em> always be invoked.
  576.      * @param errorHandler An optional error handler, which will be invoked finally, if any error occurred. The error
  577.      *        handler will receive the first error, AKA {@link Throwable}.
  578.      * @param resources The resource actions to execute. <em>All</em> resource actions will be invoked, in the given
  579.      *        order. A resource action is an instance of {@link FailableRunnable}, which will be executed.
  580.      * @see #tryWithResources(FailableRunnable, FailableRunnable...)
  581.      */
  582.     @SafeVarargs
  583.     public static void tryWithResources(final FailableRunnable<? extends Throwable> action,
  584.         final FailableConsumer<Throwable, ? extends Throwable> errorHandler,
  585.         final FailableRunnable<? extends Throwable>... resources) {
  586.         final org.apache.commons.lang3.function.FailableRunnable<?>[] fr = new org.apache.commons.lang3.function.FailableRunnable[resources.length];
  587.         Arrays.setAll(fr, i -> () -> resources[i].run());
  588.         Failable.tryWithResources(action::run, errorHandler != null ? errorHandler::accept : null, fr);
  589.     }

  590.     /**
  591.      * A simple try-with-resources implementation, that can be used, if your objects do not implement the
  592.      * {@link AutoCloseable} interface. The method executes the {@code action}. The method guarantees, that <em>all</em>
  593.      * the {@code resources} are being executed, in the given order, afterwards, and regardless of success, or failure.
  594.      * If either the original action, or any of the resource action fails, then the <em>first</em> failure (AKA
  595.      * {@link Throwable}) is rethrown. Example use:
  596.      *
  597.      * <pre>
  598.      * {@code
  599.      *     final FileInputStream fis = new FileInputStream("my.file");
  600.      *     Functions.tryWithResources(useInputStream(fis), () -> fis.close());
  601.      * }</pre>
  602.      *
  603.      * @param action The action to execute. This object <em>will</em> always be invoked.
  604.      * @param resources The resource actions to execute. <em>All</em> resource actions will be invoked, in the given
  605.      *        order. A resource action is an instance of {@link FailableRunnable}, which will be executed.
  606.      * @see #tryWithResources(FailableRunnable, FailableConsumer, FailableRunnable...)
  607.      */
  608.     @SafeVarargs
  609.     public static void tryWithResources(final FailableRunnable<? extends Throwable> action,
  610.         final FailableRunnable<? extends Throwable>... resources) {
  611.         tryWithResources(action, null, resources);
  612.     }

  613.     /**
  614.      * Constructs a new instance.
  615.      */
  616.     public Functions() {
  617.         // empty
  618.     }
  619. }