org.d2ab.iterable.chars.CharIterable Maven / Gradle / Ivy
Show all versions of sequence Show documentation
/*
* Copyright 2015 Daniel Skogquist Åborg
*
* 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 org.d2ab.iterable.chars;
import org.d2ab.function.chars.CharConsumer;
import org.d2ab.iterator.chars.ArrayCharIterator;
import org.d2ab.iterator.chars.CharIterator;
import org.d2ab.util.Pair;
import java.util.Arrays;
import java.util.Iterator;
import java.util.function.Consumer;
import java.util.stream.IntStream;
import java.util.stream.Stream;
import static java.util.Objects.requireNonNull;
@FunctionalInterface
public interface CharIterable extends Iterable {
/**
* Converts a container of some kind into a possibly once-only {@link CharIterable}.
*
* @param container the non-null container to turn into an {@link CharIterable}, can be one of {@link Iterable},
* {@link CharIterable}, {@link Iterator}, {@link CharIterator}, {@link Stream} of {@link
* Character}s, {@link IntStream}, array of {@link Character}s, or char array.
*
* @return the container as a {@link CharIterable}.
*
* @throws ClassCastException if the container is not one of the permitted classes.
*/
@SuppressWarnings("unchecked")
static CharIterable from(Object container) {
requireNonNull(container);
if (container instanceof CharIterable)
return (CharIterable) container;
else if (container instanceof Iterable)
return from((Iterable) container);
else if (container instanceof CharIterator)
return from((CharIterator) container);
else if (container instanceof Stream)
return from((Iterator) container);
else if (container instanceof IntStream)
return from((IntStream) container);
else if (container instanceof char[])
return of((char[]) container);
else if (container instanceof Character[])
return from((Character[]) container);
else if (container instanceof Pair)
return from(((Pair) container)::iterator);
else
throw new ClassCastException("Required an Iterable, CharIterable, Iterator, CharIterator, array of " +
"Character, char array, Stream of Character, or IntStream but got: " +
container.getClass());
}
static CharIterable from(Iterable iterable) {
return () -> CharIterator.from(iterable);
}
static CharIterable from(CharIterator iterator) {
return iterator.asIterable();
}
static CharIterable from(Iterator iterator) {
return () -> CharIterator.from(iterator);
}
static CharIterable from(IntStream charStream) {
return from(CharIterator.from(charStream.iterator()));
}
static CharIterable of(char... characters) {
return () -> new ArrayCharIterator(characters);
}
static CharIterable from(Character... characters) {
return from(Arrays.asList(characters));
}
static CharIterable from(Stream stream) {
return from(stream.iterator());
}
@Override
CharIterator iterator();
/**
* {@inheritDoc}
*
* If the action is an instance of {@code CharConsumer} then it is cast to {@code CharConsumer} and
* passed
* to {@link #forEachChar}; otherwise the action is adapted to an instance of {@code CharConsumer}, by boxing the
* argument of {@code CharConsumer}, and then passed to {@link #forEachChar}.
*/
@Override
default void forEach(Consumer super Character> action) {
requireNonNull(action);
forEachChar((action instanceof CharConsumer) ? (CharConsumer) action : action::accept);
}
/**
* Performs the given action for each character in this {@code Iterable} until all elements have been processed or
* the action throws an exception. Actions are performed in the order of iteration, if that order is specified.
* Exceptions thrown by the action are relayed to the caller.
*
* @param action The action to be performed for each element
*
* @throws NullPointerException if the specified action is null
*
The default implementation behaves as if:
*
{@code
* CharIterator iterator = iterator();
* while (iterator.hasNext())
* action.accept(iterator.nextChar());
* }
*/
default void forEachChar(CharConsumer action) {
requireNonNull(action);
CharIterator iterator = iterator();
while (iterator.hasNext())
action.accept(iterator.nextChar());
}
}