/*=============================================================================# # Copyright (c) 2009-2016 Stephan Wahlbrink (WalWare.de) and others. # All rights reserved. This program and the accompanying materials # are made available under the terms of either (per the licensee's choosing) # - the Eclipse Public License v1.0 # which accompanies this distribution, and is available at # http://www.eclipse.org/legal/epl-v10.html, or # - the GNU Lesser General Public License v2.1 or newer # which accompanies this distribution, and is available at # http://www.gnu.org/licenses/lgpl.html # # Contributors: # Stephan Wahlbrink - initial API and implementation #=============================================================================*/ package de.walware.rj.data.defaultImpl; import java.io.IOException; import de.walware.rj.data.RJIO; import de.walware.rj.data.RStore; public class RComplexDataBFixLongImpl extends AbstractComplexData implements ExternalizableRStore { public static final int SEGMENT_LENGTH = DEFAULT_LONG_DATA_SEGMENT_LENGTH; private final long length; protected final double[][] realValues; protected final double[][] imaginaryValues; public RComplexDataBFixLongImpl(final long length) { this.length = length; this.realValues = new2dDoubleArray(length, SEGMENT_LENGTH); this.imaginaryValues = new2dDoubleArray(length, SEGMENT_LENGTH); } public RComplexDataBFixLongImpl(final double[][] realValues, final double[][] imaginaryValues) { this.length = check2dArrayLength(realValues, SEGMENT_LENGTH); if (this.length != check2dArrayLength(imaginaryValues, SEGMENT_LENGTH)) { throw new IllegalArgumentException(); } this.realValues = realValues; this.imaginaryValues = imaginaryValues; } public RComplexDataBFixLongImpl(final RJIO io, final long length) throws IOException { this.length = length; this.realValues = new2dDoubleArray(length, SEGMENT_LENGTH); this.imaginaryValues = new2dDoubleArray(length, SEGMENT_LENGTH); for (int i = 0; i < this.realValues.length; i++) { io.readDoubleData(this.realValues[i], this.realValues[i].length); io.readDoubleData(this.imaginaryValues[i], this.imaginaryValues[i].length); } } @Override public void writeExternal(final RJIO io) throws IOException { for (int i = 0; i < this.realValues.length; i++) { io.writeDoubleData(this.realValues[i], this.realValues[i].length); io.writeDoubleData(this.imaginaryValues[i], this.imaginaryValues[i].length); } } @Override protected final boolean isStructOnly() { return false; } @Override public final long getLength() { return this.length; } @Override public boolean isNA(final int idx) { final double v = this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH]; return (Double.isNaN(v) && (int) Double.doubleToRawLongBits(v) == NA_numeric_INT_MATCH); } @Override public boolean isNA(final long idx) { final double v = this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)]; return (Double.isNaN(v) && (int) Double.doubleToRawLongBits(v) == NA_numeric_INT_MATCH); } @Override public void setNA(final int idx) { this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = NA_numeric_DOUBLE; this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = NA_numeric_DOUBLE; } @Override public void setNA(final long idx) { this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = NA_numeric_DOUBLE; this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = NA_numeric_DOUBLE; } @Override public boolean isNaN(final int idx) { final double v = this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH]; return (Double.isNaN(v) && (int) Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH); } @Override public boolean isNaN(final long idx) { final double value = this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)]; return (Double.isNaN(value) && (int) Double.doubleToRawLongBits(value) != NA_numeric_INT_MATCH); } @Override public boolean isMissing(final int idx) { return (Double.isNaN( this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] )); } @Override public boolean isMissing(final long idx) { return (Double.isNaN( this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] )); } @Override public double getCplxRe(final int idx) { return this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH]; } @Override public double getCplxRe(final long idx) { return this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)]; } @Override public double getCplxIm(final int idx) { return this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH]; } @Override public double getCplxIm(final long idx) { return this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)]; } @Override public void setCplx(final int idx, final double real, final double imaginary) { if (Double.isNaN(real) || Double.isNaN(imaginary)) { this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = NaN_numeric_DOUBLE; this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = NaN_numeric_DOUBLE; } else { this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = real; this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = imaginary; } } @Override public void setCplx(final long idx, final double real, final double imaginary) { if (Double.isNaN(real) || Double.isNaN(imaginary)) { this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = NaN_numeric_DOUBLE; this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = NaN_numeric_DOUBLE; } else { this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = real; this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = imaginary; } } @Override public void setNum(final int idx, final double real) { if (Double.isNaN(real)) { this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = NaN_numeric_DOUBLE; this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = NaN_numeric_DOUBLE; } else { this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = real; this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH] = 0.0; } } @Override public void setNum(final long idx, final double real) { if (Double.isNaN(real)) { this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = NaN_numeric_DOUBLE; this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = NaN_numeric_DOUBLE; } else { this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = real; this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)] = 0.0; } } @Override public Complex get(final int idx) { if (idx < 0 || idx >= this.length) { throw new IndexOutOfBoundsException(Long.toString(idx)); } final double v = this.realValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH]; return (!Double.isNaN(v) || (int) Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH) ? new Complex(v, this.imaginaryValues[idx / SEGMENT_LENGTH][idx % SEGMENT_LENGTH]) : null; } @Override public Complex get(final long idx) { if (idx < 0 || idx >= this.length) { throw new IndexOutOfBoundsException(Long.toString(idx)); } final double v = this.realValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)]; return (!Double.isNaN(v) || (int) Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH) ? new Complex(v, this.imaginaryValues[(int) (idx / SEGMENT_LENGTH)][(int) (idx % SEGMENT_LENGTH)]) : null; } @Override public Complex[] toArray() { final int l = checkToArrayLength(); final Complex[] array = new Complex[l]; int k = 0; for (int i = 0; i < this.realValues.length; i++, k++) { final double[] reals = this.realValues[i]; final double[] imgs = this.imaginaryValues[i]; for (int j = 0; j < reals.length; j++) { final double v = reals[j]; if (!Double.isNaN(v) || (int) Double.doubleToRawLongBits(v) != NA_numeric_INT_MATCH) { array[k] = new Complex(v, imgs[i]); } } } return array; } @Override public boolean allEqual(final RStore<?> other) { throw new UnsupportedOperationException("Not yet implemented"); } }