001/*
002 * Licensed to the Apache Software Foundation (ASF) under one or more
003 * contributor license agreements.  See the NOTICE file distributed with
004 * this work for additional information regarding copyright ownership.
005 * The ASF licenses this file to You under the Apache License, Version 2.0
006 * (the "License"); you may not use this file except in compliance with
007 * the License.  You may obtain a copy of the License at
008 *
009 *      http://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 */
017package org.apache.commons.collections4.functors;
018
019import java.io.Serializable;
020import java.util.Map;
021
022import org.apache.commons.collections4.Closure;
023import org.apache.commons.collections4.Predicate;
024
025/**
026 * Closure implementation calls the closure whose predicate returns true,
027 * like a switch statement.
028 *
029 * @since 3.0
030 * @version $Id: SwitchClosure.html 972421 2015-11-14 20:00:04Z tn $
031 */
032public class SwitchClosure<E> implements Closure<E>, Serializable {
033
034    /** Serial version UID */
035    private static final long serialVersionUID = 3518477308466486130L;
036
037    /** The tests to consider */
038    private final Predicate<? super E>[] iPredicates;
039    /** The matching closures to call */
040    private final Closure<? super E>[] iClosures;
041    /** The default closure to call if no tests match */
042    private final Closure<? super E> iDefault;
043
044    /**
045     * Factory method that performs validation and copies the parameter arrays.
046     *
047     * @param <E> the type that the closure acts on
048     * @param predicates  array of predicates, cloned, no nulls
049     * @param closures  matching array of closures, cloned, no nulls
050     * @param defaultClosure  the closure to use if no match, null means nop
051     * @return the <code>chained</code> closure
052     * @throws IllegalArgumentException if array is null
053     * @throws IllegalArgumentException if any element in the array is null
054     */
055    @SuppressWarnings("unchecked")
056    public static <E> Closure<E> switchClosure(final Predicate<? super E>[] predicates,
057                                               final Closure<? super E>[] closures,
058                                               final Closure<? super E> defaultClosure) {
059        FunctorUtils.validate(predicates);
060        FunctorUtils.validate(closures);
061        if (predicates.length != closures.length) {
062            throw new IllegalArgumentException("The predicate and closure arrays must be the same size");
063        }
064        if (predicates.length == 0) {
065            return (Closure<E>) (defaultClosure == null ? NOPClosure.<E>nopClosure(): defaultClosure);
066        }
067        return new SwitchClosure<E>(predicates, closures, defaultClosure);
068    }
069
070    /**
071     * Create a new Closure that calls one of the closures depending
072     * on the predicates.
073     * <p>
074     * The Map consists of Predicate keys and Closure values. A closure
075     * is called if its matching predicate returns true. Each predicate is evaluated
076     * until one returns true. If no predicates evaluate to true, the default
077     * closure is called. The default closure is set in the map with a
078     * null key. The ordering is that of the iterator() method on the entryset
079     * collection of the map.
080     *
081     * @param <E> the type that the closure acts on
082     * @param predicatesAndClosures  a map of predicates to closures
083     * @return the <code>switch</code> closure
084     * @throws IllegalArgumentException if the map is null
085     * @throws IllegalArgumentException if any closure in the map is null
086     * @throws ClassCastException  if the map elements are of the wrong type
087     */
088    @SuppressWarnings("unchecked")
089    public static <E> Closure<E> switchClosure(final Map<Predicate<E>, Closure<E>> predicatesAndClosures) {
090        if (predicatesAndClosures == null) {
091            throw new IllegalArgumentException("The predicate and closure map must not be null");
092        }
093        // convert to array like this to guarantee iterator() ordering
094        final Closure<? super E> defaultClosure = predicatesAndClosures.remove(null);
095        final int size = predicatesAndClosures.size();
096        if (size == 0) {
097            return (Closure<E>) (defaultClosure == null ? NOPClosure.<E>nopClosure() : defaultClosure);
098        }
099        final Closure<E>[] closures = new Closure[size];
100        final Predicate<E>[] preds = new Predicate[size];
101        int i = 0;
102        for (final Map.Entry<Predicate<E>, Closure<E>> entry : predicatesAndClosures.entrySet()) {
103            preds[i] = entry.getKey();
104            closures[i] = entry.getValue();
105            i++;
106        }
107        return new SwitchClosure<E>(false, preds, closures, defaultClosure);
108    }
109
110    /**
111     * Hidden constructor for the use by the static factory methods.
112     *
113     * @param clone  if {@code true} the input arguments will be cloned
114     * @param predicates  array of predicates, no nulls
115     * @param closures  matching array of closures, no nulls
116     * @param defaultClosure  the closure to use if no match, null means nop
117     */
118    @SuppressWarnings("unchecked")
119    private SwitchClosure(final boolean clone, final Predicate<? super E>[] predicates,
120                          final Closure<? super E>[] closures, final Closure<? super E> defaultClosure) {
121        super();
122        iPredicates = clone ? FunctorUtils.copy(predicates) : predicates;
123        iClosures = clone ? FunctorUtils.copy(closures) : closures;
124        iDefault = (Closure<? super E>) (defaultClosure == null ? NOPClosure.<E>nopClosure() : defaultClosure);
125    }
126
127    /**
128     * Constructor that performs no validation.
129     * Use <code>switchClosure</code> if you want that.
130     *
131     * @param predicates  array of predicates, cloned, no nulls
132     * @param closures  matching array of closures, cloned, no nulls
133     * @param defaultClosure  the closure to use if no match, null means nop
134     */
135    public SwitchClosure(final Predicate<? super E>[] predicates, final Closure<? super E>[] closures,
136                         final Closure<? super E> defaultClosure) {
137        this(true, predicates, closures, defaultClosure);
138    }
139
140    /**
141     * Executes the closure whose matching predicate returns true
142     *
143     * @param input  the input object
144     */
145    public void execute(final E input) {
146        for (int i = 0; i < iPredicates.length; i++) {
147            if (iPredicates[i].evaluate(input) == true) {
148                iClosures[i].execute(input);
149                return;
150            }
151        }
152        iDefault.execute(input);
153    }
154
155    /**
156     * Gets the predicates.
157     *
158     * @return a copy of the predicates
159     * @since 3.1
160     */
161    public Predicate<? super E>[] getPredicates() {
162        return FunctorUtils.<E>copy(iPredicates);
163    }
164
165    /**
166     * Gets the closures.
167     *
168     * @return a copy of the closures
169     * @since 3.1
170     */
171    public Closure<? super E>[] getClosures() {
172        return FunctorUtils.<E>copy(iClosures);
173    }
174
175    /**
176     * Gets the default closure.
177     *
178     * @return the default closure
179     * @since 3.1
180     */
181    public Closure<? super E> getDefaultClosure() {
182        return iDefault;
183    }
184
185}