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

org.ejml.equation.IntegerSequence Maven / Gradle / Ivy

/*
 * Copyright (c) 2022, Peter Abeles. All Rights Reserved.
 *
 * This file is part of Efficient Java Matrix Library (EJML).
 *
 * 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.ejml.equation;

import org.jetbrains.annotations.Nullable;

import java.util.ArrayList;
import java.util.List;

/**
 * Interface for an ordered sequence of integer values
 *
 * @author Peter Abeles
 */
public interface IntegerSequence {

    int length();

    /**
     * Specifies the maximum index of the array. If the maximum index is not known then a value < 0 is passed
     * in and an exception should be thrown if this information is required
     *
     * NOTE: This is length - 1
     *
     * @param maxIndex Largest possible value in the sequence. or < 0 if unknown
     */
    void initialize( int maxIndex );

    int next();

    boolean hasNext();

    Type getType();

    boolean requiresMaxIndex();

    enum Type {
        EXPLICIT,
        FOR,
        COMBINED,
        RANGE
    }

    /**
     * An array of integers which was explicitly specified
     */
    class Explicit implements IntegerSequence {

        List sequence = new ArrayList();
        int where;

        public Explicit( TokenList.Token start, TokenList.Token end ) {
            TokenList.Token t = start;
            while (true) {
                sequence.add((VariableInteger)t.getVariable());
                if (t == end) {
                    break;
                } else {
                    t = t.next;
                }
            }
        }

        public Explicit( TokenList.Token single ) {
            sequence.add((VariableInteger)single.getVariable());
        }

        @Override
        public int length() {
            return sequence.size();
        }

        @Override
        public void initialize( int maxIndex ) {
            where = 0;
        }

        @Override
        public int next() {
            return sequence.get(where++).value;
        }

        @Override
        public boolean hasNext() {
            return where < sequence.size();
        }

        @Override
        public Type getType() {
            return Type.EXPLICIT;
        }

        @Override
        public boolean requiresMaxIndex() {
            return false;
        }

        public List getSequence() {
            return sequence;
        }
    }

    /**
     * A sequence of integers which has been specified using a start number, end number, and step size.
     *
     * 2:3:21 = 2 5 8 11 14 17 20
     */
    class For implements IntegerSequence {

        VariableInteger start;
        @Nullable VariableInteger step;
        VariableInteger end;

        int valStart;
        int valStep;
        int valEnd;
        int where;
        int length;

        public For( TokenList.Token start, @Nullable TokenList.Token step, TokenList.Token end ) {
            this.start = (VariableInteger)start.getVariable();
            this.step = step == null ? null : (VariableInteger)step.getVariable();
            this.end = (VariableInteger)end.getVariable();
        }

        @Override
        public int length() {
            return length;
        }

        @Override
        public void initialize( int maxIndex ) {
            valStart = start.value;
            valEnd = end.value;
            if (step == null) {
                valStep = 1;
            } else {
                valStep = step.value;
            }

            if (valStep <= 0) {
                throw new IllegalArgumentException("step size must be a positive integer");
            }
            if (valEnd < valStart) {
                throw new IllegalArgumentException("end value must be >= the start value");
            }

            where = 0;
            length = (valEnd - valStart)/valStep + 1;
        }

        @Override
        public int next() {
            return valStart + valStep*where++;
        }

        @Override
        public boolean hasNext() {
            return where < length;
        }

        public int getStart() {
            return valStart;
        }

        public int getStep() {
            return valStep;
        }

        public int getEnd() {
            return valEnd;
        }

        @Override
        public Type getType() {
            return Type.FOR;
        }

        @Override
        public boolean requiresMaxIndex() {
            return false;
        }
    }

    /**
     * This is a sequence of sequences
     */
    class Combined implements IntegerSequence {

        List sequences = new ArrayList();

        int which;

        public Combined( TokenList.Token start, TokenList.Token end ) {

            TokenList.Token t = start;
            do {
                if (t.getVariable().getType() == VariableType.SCALAR) {
                    sequences.add(new Explicit(t));
                } else if (t.getVariable().getType() == VariableType.INTEGER_SEQUENCE) {
                    sequences.add(((VariableIntegerSequence)t.getVariable()).sequence);
                } else {
                    throw new RuntimeException("Unexpected token type");
                }
                t = t.next;
            } while (t != null && t.previous != end);
        }

        @Override
        public int length() {
            int total = 0;
            for (int i = 0; i < sequences.size(); i++) {
                total += sequences.get(i).length();
            }
            return total;
        }

        @Override
        public void initialize( int maxIndex ) {
            which = 0;
            for (int i = 0; i < sequences.size(); i++) {
                sequences.get(i).initialize(maxIndex);
            }
        }

        @Override
        public int next() {
            int output = sequences.get(which).next();

            if (!sequences.get(which).hasNext()) {
                which++;
            }

            return output;
        }

        @Override
        public boolean hasNext() {
            return which < sequences.size();
        }

        @Override
        public Type getType() {
            return Type.COMBINED;
        }

        @Override
        public boolean requiresMaxIndex() {
            for (int i = 0; i < sequences.size(); i++) {
                if (sequences.get(i).requiresMaxIndex())
                    return true;
            }
            return false;
        }
    }

    /**
     * A sequence of integers which has been specified using a start number, end number, and step size and uses
     * the known upper limit of the array to bound it
     *
     * Examples:
     * :
     * 2:
     * 2:3:
     */
    class Range implements IntegerSequence {

        @Nullable VariableInteger start;
        @Nullable VariableInteger step;

        int valStart;
        int valStep;
        int valEnd;
        int where;
        int length;

        public Range( @Nullable TokenList.Token start, @Nullable TokenList.Token step ) {
            this.start = start == null ? null : (VariableInteger)start.getVariable();
            this.step = step == null ? null : (VariableInteger)step.getVariable();
        }

        @Override
        public int length() {
            return length;
        }

        @Override
        public void initialize( int maxIndex ) {
            if (maxIndex < 0)
                throw new IllegalArgumentException("Range sequence being used inside an object without a known upper limit");
            valEnd = maxIndex;

            if (start != null)
                valStart = start.value;
            else
                valStart = 0;

            if (step == null) {
                valStep = 1;
            } else {
                valStep = step.value;
            }

            if (valStep <= 0) {
                throw new IllegalArgumentException("step size must be a positive integer");
            }

            where = 0;
            length = (valEnd - valStart)/valStep + 1;
        }

        @Override
        public int next() {
            return valStart + valStep*where++;
        }

        @Override
        public boolean hasNext() {
            return where < length;
        }

        public int getStart() {
            return valStart;
        }

        public int getStep() {
            return valStep;
        }

        public int getEnd() {
            return valEnd;
        }

        @Override
        public Type getType() {
            return Type.RANGE;
        }

        @Override
        public boolean requiresMaxIndex() {
            return true;
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy