All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.hazelcast.jet.Traverser Maven / Gradle / Ivy

There is a newer version: 4.5.4
Show newest version
/*
 * Copyright (c) 2008-2018, Hazelcast, Inc. All Rights Reserved.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 * http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.hazelcast.jet;

import com.hazelcast.jet.impl.util.FlatMappingTraverser;

import javax.annotation.Nonnull;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;

/**
 * Traverses a potentially infinite sequence of non-{@code null} items. Each
 * invocation of {@link #next()} consumes and returns the next item in the
 * sequence if available, or {@code null} if not. If the traverser is
 * null-terminated, getting a {@code null} means it's exhausted and
 * will keep returning {@code null} forever.
 *
 * @param  traversed item type
 */
@FunctionalInterface
public interface Traverser {

    /**
     * Returns the next item, removing it from this traverser. If no item is
     * available, returns {@code null}. If this traverser is null-terminated,
     * getting a {@code null} means it's exhausted and will keep returning
     * {@code null} forever. Otherwise, trying again later may produce one.
     */
    T next();

    /**
     * Returns a traverser that will emit the results of applying {@code
     * mapFn} to this traverser's items. If {@code mapFn} returns {@code null}
     * for an item, the returned traverser drops it and immediately moves on to
     * the next item from this traverser. This way {@code mapFn} can perform
     * filtering in addition to transformation.
     */
    @Nonnull
    default  Traverser map(@Nonnull Function mapFn) {
        return () -> {
            for (T t; (t = next()) != null;) {
                R r = mapFn.apply(t);
                if (r != null) {
                    return r;
                }
            }
            return null;
        };
    }

    /**
     * Returns a traverser that will emit the same items as this traverser, but
     * only those that pass the given predicate.
     */
    @Nonnull
    default Traverser filter(@Nonnull Predicate filterFn) {
        return () -> {
            for (T t; (t = next()) != null;) {
                if (filterFn.test(t)) {
                    return t;
                }
            }
            return null;
        };
    }

    /**
     * Returns a traverser that will apply the given mapping function to each
     * item retrieved from this traverser and emit all the items from the
     * resulting traversers, which must be null-terminated.
     */
    @Nonnull
    default  Traverser flatMap(@Nonnull Function> flatMapFn) {
        return new FlatMappingTraverser<>(this, flatMapFn);
    }

    /**
     * Returns a traverser that will emit a prefix of the original traverser,
     * up to the item for which the predicate fails (exclusive).
     */
    @Nonnull
    default Traverser takeWhile(@Nonnull Predicate pred) {
        return new Traverser() {
            boolean predicateSatisfied = true;

            @Override
            public T next() {
                if (!predicateSatisfied) {
                    return null;
                }
                T t = Traverser.this.next();
                predicateSatisfied = t == null || pred.test(t);
                return predicateSatisfied ? t : null;
            }
        };
    }

    /**
     * Returns a traverser that will emit a suffix of the original traverser,
     * starting from the item for which the predicate fails (inclusive).
     */
    @Nonnull
    default Traverser dropWhile(@Nonnull Predicate pred) {
        return new Traverser() {
            boolean predicateSatisfied = true;

            @Override
            public T next() {
                if (!predicateSatisfied) {
                    return Traverser.this.next();
                }
                for (T t; (t = Traverser.this.next()) != null; ) {
                    predicateSatisfied = pred.test(t);
                    if (!predicateSatisfied) {
                        return t;
                    }
                }
                return null;
            }
        };
    }

    /**
     * Returns a traverser that will return all the items of this traverser,
     * plus an additional item once this one returns {@code null}. After that
     * it continues forwarding the return values of this traverser. It is
     * meant to be used on finite traversers.
     * 

* Default implementations always returns a new traverser instance. If you * need to append multiple objects or use this method frequently, * {@link com.hazelcast.jet.core.AppendableTraverser} might be a better * choice. */ @Nonnull default Traverser append(@Nonnull T item) { return new Traverser() { T appendedItem = item; @Override public T next() { T t = Traverser.this.next(); if (t == null) { try { return appendedItem; } finally { appendedItem = null; } } return t; } }; } /** * Returns a traverser which prepends an additional item in front of * all the items of this traverser. */ @Nonnull default Traverser prepend(@Nonnull T item) { return new Traverser() { private boolean itemReturned; @Override public T next() { if (itemReturned) { return Traverser.this.next(); } itemReturned = true; return item; } }; } /** * Returns a traverser that will emit the same items as this traverser, * additionally passing each (non-null) item to the supplied consumer. */ @Nonnull default Traverser peek(@Nonnull Consumer action) { return () -> { T t = next(); if (t != null) { action.accept(t); } return t; }; } /** * Returns a traverser that will emit the same items as this traverser and * additionally run the supplied action the first time this traverser * returns {@code null}. */ @Nonnull default Traverser onFirstNull(@Nonnull Runnable action) { return new Traverser() { private boolean didRun; @Override public T next() { T t = Traverser.this.next(); if (t == null && !didRun) { action.run(); didRun = true; } return t; } }; } /** * Returns a traverser over the supplied arguments (or item array). * * @param items the items to traverse over * @param type of the items */ @SafeVarargs static Traverser over(T... items) { return Traversers.traverseArray(items); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy