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: 5.5.0
Show newest version
/*
 * Copyright (c) 2008-2021, 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.core.AppendableTraverser;
import com.hazelcast.jet.impl.util.FlatMappingTraverser;

import javax.annotation.CheckReturnValue;
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.
 * 

* * All transformation methods ({@code map()}, {@code append()}, {@code peek()}, * ...) are allowed to either return a new instance or to modify {@code this} * instance. For correct functionality, you must always use the returned value * and stop using the original instance. * * @param traversed item type * * @since Jet 3.0 */ @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 @CheckReturnValue 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 @CheckReturnValue 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. *

* The function must not return null traverser, but can return an {@linkplain * Traversers#empty() empty traverser}. */ @Nonnull @CheckReturnValue 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 @CheckReturnValue default Traverser takeWhile(@Nonnull Predicate predicate) { return new Traverser() { boolean predicateSatisfied = true; @Override public T next() { if (!predicateSatisfied) { return null; } T t = Traverser.this.next(); predicateSatisfied = t == null || predicate.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 @CheckReturnValue default Traverser dropWhile(@Nonnull Predicate predicate) { 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 = predicate.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 AppendableTraverser} might be a better choice. */ @Nonnull @CheckReturnValue 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 @CheckReturnValue 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 @CheckReturnValue 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 @CheckReturnValue 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; } }; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy