/* Copyright � 1999 CERN - European Organization for Nuclear Research. Permission to use, copy, modify, distribute and sell this software and its documentation for any purpose is hereby granted without fee, provided that the above copyright notice appear in all copies and that both that copyright notice and this permission notice appear in supporting documentation. CERN makes no representations about the suitability of this software for any purpose. It is provided "as is" without expressed or implied warranty. */ package cern.colt.matrix.impl; /** Abstract base class for 1-d matrices (aka <i>vectors</i>) holding objects or primitive data types such as <code>int</code>, <code>double</code>, etc. First see the <a href="package-summary.html">package summary</a> and javadoc <a href="package-tree.html">tree view</a> to get the broad picture. <p> <b>Note that this implementation is not synchronized.</b> @author wolfgang.hoschek@cern.ch @version 1.0, 09/24/99 */ public abstract class AbstractMatrix1D extends AbstractMatrix { /** the number of cells this matrix (view) has */ protected int size; /** the index of the first element */ protected int zero; /** the number of indexes between any two elements, i.e. <tt>index(i+1) - index(i)</tt>. */ protected int stride; /** * Indicates non-flipped state (flip==1) or flipped state (flip==-1). * see _setFlip() for further info. */ //protected int flip; /** * Indicates non-flipped state or flipped state. * see _setFlip() for further info. */ //protected int flipMask; // this.isNoView implies: offset==0, stride==1 /** * Makes this class non instantiable, but still let's others inherit from it. */ protected AbstractMatrix1D() {} /** * Returns the position of the given absolute rank within the (virtual or non-virtual) internal 1-dimensional array. * Default implementation. Override, if necessary. * * @param rank the absolute rank of the element. * @return the position. */ protected int _offset(int absRank) { return absRank; } /** * Returns the absolute rank of the given relative rank. * * @param rank the relative rank of the element. * @return the absolute rank of the element. */ protected int _rank(int rank) { return zero + rank*stride; //return zero + ((rank+flipMask)^flipMask); //return zero + rank*flip; // slower } /** * Sanity check for operations requiring an index to be within bounds. * @throws IndexOutOfBoundsException if <tt>index < 0 || index >= size()</tt>. */ protected void checkIndex(int index) { if (index < 0 || index >= size) throw new IndexOutOfBoundsException("Attempted to access "+toStringShort()+" at index="+index); } /** * Checks whether indexes are legal and throws an exception, if necessary. * @throws IndexOutOfBoundsException if <tt>! (0 <= indexes[i] < size())</tt> for any i=0..indexes.length()-1. */ protected void checkIndexes(int[] indexes) { for (int i=indexes.length; --i >= 0; ) { int index = indexes[i]; if (index < 0 || index >= size) checkIndex(index); } } /** * Checks whether the receiver contains the given range and throws an exception, if necessary. * @throws IndexOutOfBoundsException if <tt>index<0 || index+width>size()</tt>. */ protected void checkRange(int index, int width) { if (index<0 || index+width>size) throw new IndexOutOfBoundsException("index: "+index+", width: "+width+", size="+size); } /** * Sanity check for operations requiring two matrices with the same size. * @throws IllegalArgumentException if <tt>size() != B.size()</tt>. */ protected void checkSize(double[] B) { if (size != B.length) throw new IllegalArgumentException("Incompatible sizes: "+toStringShort()+" and "+B.length); } /** * Sanity check for operations requiring two matrices with the same size. * @throws IllegalArgumentException if <tt>size() != B.size()</tt>. */ public void checkSize(AbstractMatrix1D B) { if (size != B.size) throw new IllegalArgumentException("Incompatible sizes: "+toStringShort()+" and "+B.toStringShort()); } /** * Returns the position of the element with the given relative rank within the (virtual or non-virtual) internal 1-dimensional array. * You may want to override this method for performance. * * @param rank the rank of the element. */ protected int index(int rank) { return _offset(_rank(rank)); } /** * Sets up a matrix with a given number of cells. * @param size the number of cells the matrix shall have. * @throws IllegalArgumentException if <tt>size<0</tt>. */ protected void setUp(int size) { setUp(size,0,1); } /** * Sets up a matrix with the given parameters. * @param size the number of elements the matrix shall have. * @param zero the index of the first element. * @param stride the number of indexes between any two elements, i.e. <tt>index(i+1)-index(i)</tt>. * @throws IllegalArgumentException if <tt>size<0</tt>. */ protected void setUp(int size, int zero, int stride) { if (size<0) throw new IllegalArgumentException("negative size"); this.size = size; this.zero = zero; this.stride = stride; this.isNoView = true; } /** * Returns the number of cells. */ public int size() { return size; } /** * Returns the stride of the given dimension (axis, rank). * * @dimension the index of the dimension. * @return the stride in the given dimension. * @throws IllegalArgumentException if <tt>dimension != 0</tt>. */ protected int stride(int dimension) { if (dimension != 0) throw new IllegalArgumentException("invalid dimension: "+dimension+"used to access"+toStringShort()); return this.stride; } /** * Returns a string representation of the receiver's shape. */ public String toStringShort() { return AbstractFormatter.shape(this); } /** Self modifying version of viewFlip(). What used to be index <tt>0</tt> is now index <tt>size()-1</tt>, ..., what used to be index <tt>size()-1</tt> is now index <tt>0</tt>. */ protected AbstractMatrix1D vFlip() { if (size>0) { this.zero += (this.size-1)*this.stride; this.stride = - this.stride; this.isNoView = false; } return this; } /** Self modifying version of viewPart(). @throws IndexOutOfBoundsException if <tt>index<0 || index+width>size()</tt>. */ protected AbstractMatrix1D vPart(int index, int width) { checkRange(index,width); this.zero += this.stride * index; this.size = width; this.isNoView = false; return this; } /** Self modifying version of viewStrides(). @throws IndexOutOfBoundsException if <tt>stride <= 0</tt>. */ protected AbstractMatrix1D vStrides(int stride) { if (stride<=0) throw new IndexOutOfBoundsException("illegal stride: "+stride); this.stride *= stride; if (this.size!=0) this.size = (this.size-1)/stride +1; this.isNoView = false; return this; } }