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

pl.edu.icm.jlargearrays.LogicLargeArray Maven / Gradle / Ivy

There is a newer version: 1.6
Show newest version
/* ***** BEGIN LICENSE BLOCK *****
 * JLargeArrays
 * Copyright (C) 2013 onward University of Warsaw, ICM
 * All rights reserved.
 * 
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 * 
 * 1. Redistributions of source code must retain the above copyright notice, this
 *    list of conditions and the following disclaimer. 
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 *    this list of conditions and the following disclaimer in the documentation
 *    and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 *
 * ***** END LICENSE BLOCK ***** */
package pl.edu.icm.jlargearrays;

import java.util.concurrent.ExecutionException;
import java.util.concurrent.Future;
import sun.misc.Cleaner;
import org.apache.commons.math3.util.FastMath;

/**
 *
 * An array of bits (0 and 1) that can store up to 263 elements.
 *
 * @author Piotr Wendykier ([email protected])
 */
public class LogicLargeArray extends LargeArray
{

    private static final long serialVersionUID = 3135411647668758832L;
    private byte[] data;

    /**
     * Creates new instance of this class.
     *
     * @param length number of elements
     */
    public LogicLargeArray(long length)
    {
        this(length, true);
    }

    /**
     * Creates new instance of this class.
     *
     * @param length           number of elements
     * @param zeroNativeMemory if true, then the native memory is zeroed.
     */
    public LogicLargeArray(long length, boolean zeroNativeMemory)
    {
        this.type = LargeArrayType.LOGIC;
        this.sizeof = 1;
        if (length <= 0) {
            throw new IllegalArgumentException(length + " is not a positive long value");
        }
        this.length = length;
        if (length > getMaxSizeOf32bitArray()) {
            this.ptr = LargeArrayUtils.UNSAFE.allocateMemory(this.length * this.sizeof);
            if (zeroNativeMemory) {
                zeroNativeMemory(length);
            }
            Cleaner.create(this, new LargeArray.Deallocator(this.ptr, this.length, this.sizeof));
            MemoryCounter.increaseCounter(this.length * this.sizeof);
        } else {
            data = new byte[(int) length];
        }
    }

    /**
     * Creates a constant array.
     * 

* @param length number of elements * @param constantValue value */ public LogicLargeArray(long length, byte constantValue) { this.type = LargeArrayType.LOGIC; this.sizeof = 1; if (length <= 0) { throw new IllegalArgumentException(length + " is not a positive long value"); } this.length = length; this.isConstant = true; this.data = new byte[]{constantValue != 0 ? (byte) 1 : (byte) 0}; } /** * Creates new instance of this class. * * @param data data array, this reference is used internally. */ public LogicLargeArray(byte[] data) { this.type = LargeArrayType.LOGIC; this.sizeof = 1; this.length = data.length; for (int i = 0; i < data.length; i++) { if (data[i] != 0 && data[i] != 1) { throw new IllegalArgumentException("The array contans values different than 0 and 1."); } } this.data = data; } /** * Creates new instance of this class. * * @param data data array, this reference is not used internally. */ public LogicLargeArray(boolean[] data) { this.type = LargeArrayType.LOGIC; this.sizeof = 1; this.length = data.length; this.data = new byte[data.length]; for (int i = 0; i < data.length; i++) { this.data[i] = (data[i] == true ? (byte) 1 : (byte) 0); } } /** * Returns a deep copy of this instance. (The elements themselves are copied.) * * @return a clone of this instance */ @Override public LogicLargeArray clone() { if (isConstant) { return new LogicLargeArray(length, getByte(0)); } else { LogicLargeArray v = new LogicLargeArray(length, false); LargeArrayUtils.arraycopy(this, 0, v, 0, length); return v; } } @Override public boolean equals(Object o) { if (super.equals(o)) { LogicLargeArray la = (LogicLargeArray) o; return this.data == la.data; } return false; } @Override public int hashCode() { return 29 * super.hashCode() + (this.data != null ? this.data.hashCode() : 0); } @Override public final Byte get(long i) { return getByte(i); } @Override public final Byte getFromNative(long i) { return LargeArrayUtils.UNSAFE.getByte(ptr + i); } @Override public final boolean getBoolean(long i) { if (ptr != 0) { return (LargeArrayUtils.UNSAFE.getByte(ptr + i)) != 0; } else if (isConstant) { return data[0] != 0; } else { return data[(int) i] != 0; } } @Override public final byte getByte(long i) { if (ptr != 0) { return LargeArrayUtils.UNSAFE.getByte(ptr + i); } else if (isConstant) { return data[0]; } else { return data[(int) i]; } } @Override public final short getUnsignedByte(long i) { return getByte(i); } @Override public final short getShort(long i) { if (ptr != 0) { return (short) (LargeArrayUtils.UNSAFE.getByte(ptr + i)); } else if (isConstant) { return data[0]; } else { return (short) data[(int) i]; } } @Override public final int getInt(long i) { if (ptr != 0) { return (int) (LargeArrayUtils.UNSAFE.getByte(ptr + i)); } else if (isConstant) { return data[0]; } else { return (int) data[(int) i]; } } @Override public final long getLong(long i) { if (ptr != 0) { return (long) (LargeArrayUtils.UNSAFE.getByte(ptr + i)); } else if (isConstant) { return data[0]; } else { return (long) data[(int) i]; } } @Override public final float getFloat(long i) { if (ptr != 0) { return (float) (LargeArrayUtils.UNSAFE.getByte(ptr + i)); } else if (isConstant) { return data[0]; } else { return (float) data[(int) i]; } } @Override public final double getDouble(long i) { if (ptr != 0) { return (double) LargeArrayUtils.UNSAFE.getByte(ptr + i); } else if (isConstant) { return data[0]; } else { return (double) data[(int) i]; } } @Override public final boolean[] getBooleanData(boolean[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { boolean[] out; if (a != null && a.length >= len) { out = a; } else { out = new boolean[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { byte v = LargeArrayUtils.UNSAFE.getByte(ptr + i); out[idx++] = v == 1; } } else if (isConstant) { boolean elem = data[0] != 0; for (long i = startPos; i < endPos; i += step) { out[idx++] = elem; } } else { for (long i = startPos; i < endPos; i += step) { byte v = data[(int) i]; out[idx++] = v != 0; } } return out; } } @Override public final boolean[] getBooleanData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; boolean[] out = new boolean[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { byte v = LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); out[i] = v != 0; } } else if (isConstant) { boolean elem = data[0] != 0; for (int i = 0; i < length; i++) { out[i] = elem; } } else { for (int i = 0; i < length; i++) { out[i] = data[i] != 0; } } return out; } @Override public final byte[] getData() { return data; } @Override public final byte[] getByteData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; byte[] out = new byte[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { out[i] = LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); } } else if (isConstant) { byte elem = data[0]; for (int i = 0; i < length; i++) { out[i] = elem; } } else { System.arraycopy(data, 0, out, 0, (int) length); } return out; } @Override public final byte[] getByteData(byte[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { byte[] out; if (a != null && a.length >= len) { out = a; } else { out = new byte[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { out[idx++] = LargeArrayUtils.UNSAFE.getByte(ptr + i); } } else if (isConstant) { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[0]; } } else { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[(int) i]; } } return out; } } @Override public final short[] getShortData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; short[] out = new short[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { out[i] = (short) LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); } } else if (isConstant) { byte elem = data[0]; for (int i = 0; i < length; i++) { out[i] = (short) elem; } } else { for (int i = 0; i < length; i++) { out[i] = (short) data[i]; } } return out; } @Override public final short[] getShortData(short[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { short[] out; if (a != null && a.length >= len) { out = a; } else { out = new short[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { out[idx++] = (short) LargeArrayUtils.UNSAFE.getByte(ptr + i); } } else if (isConstant) { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[0]; } } else { for (long i = startPos; i < endPos; i += step) { out[idx++] = (short) data[(int) i]; } } return out; } } @Override public final int[] getIntData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; int[] out = new int[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { out[i] = (int) LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); } } else if (isConstant) { byte elem = data[0]; for (int i = 0; i < length; i++) { out[i] = (int) elem; } } else { for (int i = 0; i < length; i++) { out[i] = (int) data[i]; } } return out; } @Override public final int[] getIntData(int[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { int[] out; if (a != null && a.length >= len) { out = a; } else { out = new int[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { out[idx++] = (int) LargeArrayUtils.UNSAFE.getByte(ptr + i); } } else if (isConstant) { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[0]; } } else { for (long i = startPos; i < endPos; i += step) { out[idx++] = (int) data[(int) i]; } } return out; } } @Override public final long[] getLongData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; long[] out = new long[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { out[i] = (long) LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); } } else if (isConstant) { byte elem = data[0]; for (int i = 0; i < length; i++) { out[i] = (long) elem; } } else { for (int i = 0; i < length; i++) { out[i] = (long) data[i]; } } return out; } @Override public final long[] getLongData(long[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { long[] out; if (a != null && a.length >= len) { out = a; } else { out = new long[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { out[idx++] = (long) LargeArrayUtils.UNSAFE.getByte(ptr + i); } } else if (isConstant) { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[0]; } } else { for (long i = startPos; i < endPos; i += step) { out[idx++] = (long) data[(int) i]; } } return out; } } @Override public final float[] getFloatData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; float[] out = new float[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { out[i] = (float) LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); } } else if (isConstant) { float elem = (float) data[0]; for (int i = 0; i < length; i++) { out[i] = elem; } } else { for (int i = 0; i < length; i++) { out[i] = (float) data[i]; } } return out; } @Override public final float[] getFloatData(float[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { float[] out; if (a != null && a.length >= len) { out = a; } else { out = new float[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { out[idx++] = (float) LargeArrayUtils.UNSAFE.getByte(ptr + i); } } else if (isConstant) { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[0]; } } else { for (long i = startPos; i < endPos; i += step) { out[idx++] = (float) data[(int) i]; } } return out; } } @Override public final double[] getDoubleData() { if (length > LargeArray.LARGEST_SUBARRAY) return null; double[] out = new double[(int) length]; if (ptr != 0) { for (int i = 0; i < length; i++) { out[i] = (double) LargeArrayUtils.UNSAFE.getByte(ptr + sizeof * i); } } else if (isConstant) { double elem = (double) data[0]; for (int i = 0; i < length; i++) { out[i] = elem; } } else { for (int i = 0; i < length; i++) { out[i] = (double) data[i]; } } return out; } @Override public final double[] getDoubleData(double[] a, long startPos, long endPos, long step) { if (startPos < 0 || startPos >= length) { throw new ArrayIndexOutOfBoundsException("startPos < 0 || startPos >= length"); } if (endPos < 0 || endPos > length || endPos < startPos) { throw new ArrayIndexOutOfBoundsException("endPos < 0 || endPos > length || endPos < startPos"); } if (step < 1) { throw new IllegalArgumentException("step < 1"); } long len = (long) FastMath.ceil((endPos - startPos) / (double) step); if (len > LargeArray.LARGEST_SUBARRAY) { return null; } else { double[] out; if (a != null && a.length >= len) { out = a; } else { out = new double[(int) len]; } int idx = 0; if (ptr != 0) { for (long i = startPos; i < endPos; i += step) { out[idx++] = (double) LargeArrayUtils.UNSAFE.getByte(ptr + i); } } else if (isConstant) { for (long i = startPos; i < endPos; i += step) { out[idx++] = data[0]; } } else { for (long i = startPos; i < endPos; i += step) { out[idx++] = (double) data[(int) i]; } } return out; } } @Override public final void setToNative(long i, Object value) { LargeArrayUtils.UNSAFE.putByte(ptr + i, (Byte) value); } @Override public final void setBoolean(long i, boolean value) { if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, value == true ? (byte) 1 : (byte) 0); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = value == true ? (byte) 1 : (byte) 0; } } @Override public final void setByte(long i, byte value) { if (value < 0 || value > 1) { throw new IllegalArgumentException("The value has to be 0 or 1."); } if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, value); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = value; } } @Override public final void setUnsignedByte(long i, short value) { setShort(i, value); } @Override public final void setShort(long i, short value) { if (value < 0 || value > 1) { throw new IllegalArgumentException("The value has to be 0 or 1."); } if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, (byte) value); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = (byte) value; } } @Override public final void setInt(long i, int value) { if (value < 0 || value > 1) { throw new IllegalArgumentException("The value has to be 0 or 1."); } if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, (byte) value); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = (byte) value; } } @Override public final void setLong(long i, long value) { if (value < 0 || value > 1) { throw new IllegalArgumentException("The value has to be 0 or 1."); } if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, (byte) value); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = (byte) value; } } @Override public final void setFloat(long i, float value) { if (value != 0.0 && value != 1.0) { throw new IllegalArgumentException("The value has to be 0 or 1."); } if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, (byte) value); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = (byte) value; } } @Override public final void setDouble(long i, double value) { if (value != 0.0 && value != 1.0) { throw new IllegalArgumentException("The value has to be 0 or 1."); } if (ptr != 0) { LargeArrayUtils.UNSAFE.putByte(ptr + i, (byte) value); } else { if (isConstant) { throw new IllegalAccessError("Constant arrays cannot be modified."); } data[(int) i] = (byte) value; } } /** * Performs binary and operation using this array and the input array *

* @param array input array *

* @return (this and array) */ public LogicLargeArray and(final LogicLargeArray array) { if (array == null || array.length() != length) { throw new IllegalArgumentException("array == null || array.length() != length"); } final LogicLargeArray out = new LogicLargeArray(array.length(), false); final int nthreads = (int) FastMath.min(length, ConcurrencyUtils.getNumberOfThreads()); if (nthreads <= 2 || length < ConcurrencyUtils.getConcurrentThreshold()) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) & array.getByte(i))); } } else { final long k = length / nthreads; final Future[] threads = new Future[nthreads]; for (int j = 0; j < nthreads; j++) { final long firstIdx = j * k; final long lastIdx = (j == nthreads - 1) ? length : firstIdx + k; threads[j] = ConcurrencyUtils.submit(new Runnable() { @Override public void run() { for (long k = firstIdx; k < lastIdx; k++) { out.setByte(k, (byte) (getByte(k) & array.getByte(k))); } } }); } try { ConcurrencyUtils.waitForCompletion(threads); } catch (final InterruptedException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) & array.getByte(i))); } } catch (ExecutionException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) & array.getByte(i))); } } } return out; } /** * Performs binary or operation using this array and the input array *

* @param array input array *

* @return (this or array) */ public LogicLargeArray or(final LogicLargeArray array) { if (array == null || array.length() != length) { throw new IllegalArgumentException("array == null || array.length() != length"); } final LogicLargeArray out = new LogicLargeArray(array.length(), false); final int nthreads = (int) FastMath.min(length, ConcurrencyUtils.getNumberOfThreads()); if (nthreads <= 2 || length < ConcurrencyUtils.getConcurrentThreshold()) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) | array.getByte(i))); } } else { final long k = length / nthreads; final Future[] threads = new Future[nthreads]; for (int j = 0; j < nthreads; j++) { final long firstIdx = j * k; final long lastIdx = (j == nthreads - 1) ? length : firstIdx + k; threads[j] = ConcurrencyUtils.submit(new Runnable() { @Override public void run() { for (long k = firstIdx; k < lastIdx; k++) { out.setByte(k, (byte) (getByte(k) | array.getByte(k))); } } }); } try { ConcurrencyUtils.waitForCompletion(threads); } catch (final InterruptedException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) | array.getByte(i))); } } catch (ExecutionException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) | array.getByte(i))); } } } return out; } /** * Performs binary and operation using this array and the input array *

* @param array input array *

* @return (this xor array) */ public LogicLargeArray xor(final LogicLargeArray array) { if (array == null || array.length() != length) { throw new IllegalArgumentException("array == null || array.length() != length"); } final LogicLargeArray out = new LogicLargeArray(array.length(), false); final int nthreads = (int) FastMath.min(length, ConcurrencyUtils.getNumberOfThreads()); if (nthreads <= 2 || length < ConcurrencyUtils.getConcurrentThreshold()) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) ^ array.getByte(i))); } } else { final long k = length / nthreads; final Future[] threads = new Future[nthreads]; for (int j = 0; j < nthreads; j++) { final long firstIdx = j * k; final long lastIdx = (j == nthreads - 1) ? length : firstIdx + k; threads[j] = ConcurrencyUtils.submit(new Runnable() { @Override public void run() { for (long k = firstIdx; k < lastIdx; k++) { out.setByte(k, (byte) (getByte(k) ^ array.getByte(k))); } } }); } try { ConcurrencyUtils.waitForCompletion(threads); } catch (final InterruptedException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) ^ array.getByte(i))); } } catch (ExecutionException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (getByte(i) ^ array.getByte(i))); } } } return out; } /** * Performs binary negation of this array *

* @return (not this) */ public LogicLargeArray not() { final LogicLargeArray out = new LogicLargeArray(length, false); final int nthreads = (int) FastMath.min(length, ConcurrencyUtils.getNumberOfThreads()); if (nthreads <= 2 || length < ConcurrencyUtils.getConcurrentThreshold()) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (1 - getByte(i))); } } else { final long k = length / nthreads; final Future[] threads = new Future[nthreads]; for (int j = 0; j < nthreads; j++) { final long firstIdx = j * k; final long lastIdx = (j == nthreads - 1) ? length : firstIdx + k; threads[j] = ConcurrencyUtils.submit(new Runnable() { @Override public void run() { for (long k = firstIdx; k < lastIdx; k++) { out.setByte(k, (byte) (1 - getByte(k))); } } }); } try { ConcurrencyUtils.waitForCompletion(threads); } catch (final InterruptedException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (1 - getByte(i))); } } catch (ExecutionException ex) { for (long i = 0; i < length; i++) { out.setByte(i, (byte) (1 - getByte(i))); } } } return out; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy