parsii.tokenizer.Lookahead Maven / Gradle / Ivy
/*
* Made with all the love in the world
* by scireum in Remshalden, Germany
*
* Copyright by scireum GmbH
* http://www.scireum.de - [email protected]
*/
package parsii.tokenizer;
import java.util.ArrayList;
import java.util.List;
/**
* Abstract data structure for providing streams of items with a lookahead.
*
* Items provided by subclasses of this are processed one after each other. However, using {@link #next()} or
* {@link #next(int)} one can peek at upcoming items without consuming the current one.
*
* @param the type of the elements kept in the stream
*/
public abstract class Lookahead {
/**
* Internal buffer containing items which where already created due to lookaheads.
*/
protected List itemBuffer = new ArrayList<>();
/**
* Determines if the end of the underlying data source has been reached.
*/
protected boolean endReached = false;
/**
* Used to collect problems which occurred when processing the input. This is used instead of classic exceptions,
* so that errors can be recovered and we can continue to process to input to check for further errors or
* problems.
*/
protected List problemCollector = new ArrayList<>();
/**
* Once the end of the underlying input was reached, an end of input indicator is created and constantly returned
* for all calls of current and next.
*/
protected T endOfInputIndicator;
/**
* Returns the item the stream is currently pointing at.
*
* This method does not change the internal state. Therefore it can be called several times and will always
* return the same result.
*
* @return the item the stream is currently pointing at.
*/
public T current() {
return next(0);
}
/**
* Returns the next item after the current one in the stream.
*
* This method does not change the internal state. Therefore it can be called several times and will always
* return the same result.
*
* @return the next item in the stream. This will be the current item, after a call to {@link #consume()}
*/
public T next() {
return next(1);
}
/**
* Returns the next n-th item in the stream.
*
* Calling this method with 0 as parameter, will return the current item. Calling it with 1 will return the
* same item as a call to next().
*
* This method does not change the internal state. Therefore it can be called several times and will always
* return the same result.
*
* @param offset the number of items to skip
* @return the n-th item in the stream
*/
public T next(int offset) {
if (offset < 0) {
throw new IllegalArgumentException("offset < 0");
}
while (itemBuffer.size() <= offset && !endReached) {
T item = fetch();
if (item != null) {
itemBuffer.add(item);
} else {
endReached = true;
}
}
if (offset >= itemBuffer.size()) {
if (endOfInputIndicator == null) {
endOfInputIndicator = endOfInput();
}
return endOfInputIndicator;
} else {
return itemBuffer.get(offset);
}
}
/**
* Creates the end of input indicator item.
*
* This method will be only called once, as the indicator is cached.
*
* @return a special item which marks the end of the input
*/
protected abstract T endOfInput();
/**
* Fetches the next item from the stream.
*
* @return the next item in the stream or null to indicate that the end was reached
*/
protected abstract T fetch();
/**
* Removes and returns the current item from the stream.
*
After this method was called, all calls to {@link #current()} will then return the item, which
* was previously returned by {@link #next()}
*
* @return the item which is being removed from the stream
*/
public T consume() {
T result = current();
consume(1);
return result;
}
/**
* Consumes (removes) numberOfItems at once.
*
* Removes the given number of items from the stream.
*
* @param numberOfItems the number of items to remove
*/
public void consume(int numberOfItems) {
if (numberOfItems < 0) {
throw new IllegalArgumentException("numberOfItems < 0");
}
while (numberOfItems-- > 0) {
if (!itemBuffer.isEmpty()) {
itemBuffer.remove(0);
} else {
if (endReached) {
return;
}
T item = fetch();
if (item == null) {
endReached = true;
}
}
}
}
/**
* Provides access to the problem collector used by this instance.
*
* @return the problem collector used by this class. This returns the internally used list, therefore it should be
* treat appropriately
*/
public List getProblemCollector() {
return problemCollector;
}
/**
* Installs the given problem collector.
*
* @param problemCollector the problem collector to be from now on
*/
public void setProblemCollector(List problemCollector) {
this.problemCollector = problemCollector;
}
}