functionalj.stream.doublestream.GrowOnlyDoubleArray Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of functionalj-core Show documentation
Show all versions of functionalj-core Show documentation
The module for FunctionalJ Core.
// ============================================================================
// Copyright (c) 2017-2021 Nawapunth Manusitthipol (NawaMan - http://nawaman.net).
// ----------------------------------------------------------------------------
// MIT License
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
// ============================================================================
package functionalj.stream.doublestream;
import java.util.OptionalDouble;
import functionalj.list.doublelist.DoubleFuncList;
import functionalj.list.doublelist.ImmutableDoubleFuncList;
import functionalj.stream.StreamPlus;
// TODO - This is NOT thread safe (not even try to be).
public final class GrowOnlyDoubleArray {
private static int ARRAY_COUNT = 8;
private static int ARRAY_LENGTH = 100;
private double[][] arrays;
private int length = 0;
public GrowOnlyDoubleArray() {
arrays = new double[ARRAY_COUNT][];
}
public GrowOnlyDoubleArray(double ... values) {
int actualCount = (int)Math.ceil(1.0 * values.length / ARRAY_LENGTH);
int arrayCount = Math.max(actualCount, ARRAY_COUNT);
arrays = new double[arrayCount][];
length = values.length;
int offset = 0;
for (int i = 0; i < actualCount; i++) {
arrays[i] = new double[ARRAY_LENGTH];
int eachLength = Math.min(ARRAY_LENGTH, length - offset);
System.arraycopy(values, offset, arrays[i], 0, eachLength);
offset = offset + ARRAY_LENGTH;
}
}
public void add(double i) {
int next = length;
int aIndex = next / ARRAY_LENGTH;
int residue = next % ARRAY_LENGTH;
if (aIndex >= arrays.length) {
double[][] newArrays = new double[arrays.length + ARRAY_COUNT][];
System.arraycopy(arrays, 0, newArrays, 0, arrays.length);
arrays = newArrays;
}
if (arrays[aIndex] == null) {
arrays[aIndex] = new double[ARRAY_LENGTH];
}
arrays[aIndex][residue] = i;
length++;
}
public int length() {
return length;
}
public boolean isEmpty() {
return length == 0;
}
public DoubleStreamPlus stream() {
int aCount = length / ARRAY_LENGTH;
int residue = length % ARRAY_LENGTH;
DoubleStreamPlus head = StreamPlus.of(arrays).limit(aCount).flatMapToDouble(a -> DoubleStreamPlus.of(a));
DoubleStreamPlus tail = ((aCount >= arrays.length) || (arrays[aCount] == null))
? DoubleStreamPlus.empty()
: DoubleStreamPlus.of(arrays[aCount]).limit(residue);
DoubleStreamPlus total = head.appendWith(tail);
return total;
}
public DoubleFuncList toFuncList() {
return ImmutableDoubleFuncList.from(stream());
}
public double[] toArray() {
return stream().toArray();
}
public double get(int i) {
if (i < 0 || i >= length)
throw new ArrayIndexOutOfBoundsException(i);
int aIndex = i / ARRAY_LENGTH;
int residue = i % ARRAY_LENGTH;
return arrays[aIndex][residue];
}
public OptionalDouble at(int i) {
if (i < 0 || i >= length)
return OptionalDouble.empty();
int aIndex = i / ARRAY_LENGTH;
int residue = i % ARRAY_LENGTH;
return OptionalDouble.of(arrays[aIndex][residue]);
}
public String toString() {
return stream().toListString();
}
public int hashCode() {
return Double.hashCode(stream().reduce(43, (p, c)-> p*43 + c));
}
public boolean equals(GrowOnlyDoubleArray array) {
int score = (int)stream().zipWith(array.stream(), (a,b) -> a == b ? 1 : 0).takeUntil(i -> i == 0).sum();
return score == length;
}
}