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

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; } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy