org.roaringbitmap.RoaringArray Maven / Gradle / Ivy
The newest version!
/*
* (c) the authors Licensed under the Apache License, Version 2.0.
*/
package org.roaringbitmap;
import java.io.*;
import java.nio.ByteBuffer;
import java.nio.CharBuffer;
import java.nio.LongBuffer;
import java.util.Arrays;
import java.util.NoSuchElementException;
import static java.nio.ByteOrder.LITTLE_ENDIAN;
/**
* Specialized array to store the containers used by a RoaringBitmap. This is not meant to be used
* by end users.
*/
public final class RoaringArray implements Cloneable, Externalizable, AppendableStorage {
private static final char SERIAL_COOKIE_NO_RUNCONTAINER = 12346;
private static final char SERIAL_COOKIE = 12347;
private static final int NO_OFFSET_THRESHOLD = 4;
// bumped serialVersionUID with runcontainers, so default serialization
// will not work...
private static final long serialVersionUID = 8L;
static final int INITIAL_CAPACITY = 4;
char[] keys = null;
Container[] values = null;
int size = 0;
protected RoaringArray() {
this(INITIAL_CAPACITY);
}
RoaringArray(int initialCapacity) {
this(new char[initialCapacity], new Container[initialCapacity], 0);
}
RoaringArray(char[] keys, Container[] values, int size) {
this.keys = keys;
this.values = values;
this.size = size;
}
/**
* Find the smallest integer index larger than pos such that array[index].key>=x. If none can
* be found, return size. Based on code by O. Kaser.
*
* @param x minimal value
* @param pos index to exceed
* @return the smallest index greater than pos such that array[index].key is at least as large as
* min, or size if it is not possible.
*/
protected int advanceUntil(char x, int pos) {
int lower = pos + 1;
// special handling for a possibly common sequential case
if (lower >= size || keys[lower] >= x) {
return lower;
}
int spansize = 1; // could set larger
// bootstrap an upper limit
while (lower + spansize < size
&& (keys[lower + spansize]) < x) {
spansize *= 2; // hoping for compiler will reduce to shift
}
int upper = (lower + spansize < size) ? lower + spansize : size - 1;
// maybe we are lucky (could be common case when the seek ahead
// expected to be small and sequential will otherwise make us look bad)
if (keys[upper] == x) {
return upper;
}
if (keys[upper] < x) {// means array has no item key >=
// x
return size;
}
// we know that the next-smallest span was too small
lower += spansize / 2;
// else begin binary search
// invariant: array[lower]x
while (lower + 1 != upper) {
int mid = (lower + upper) / 2;
if (keys[mid] == x) {
return mid;
} else if (keys[mid] < x) {
lower = mid;
} else {
upper = mid;
}
}
return upper;
}
@Override
public void append(char key, Container value) {
if (size > 0 && key < keys[size - 1]) {
throw new IllegalArgumentException("append only: "
+ (key) + " < " + (keys[size - 1]));
}
extendArray(1);
keys[size] = key;
values[size] = value;
size++;
}
void append(RoaringArray roaringArray) {
assert size == 0 || roaringArray.size == 0
|| keys[size - 1] < roaringArray.keys[0];
if (roaringArray.size != 0 && size != 0) {
keys = Arrays.copyOf(keys, size + roaringArray.size);
values = Arrays.copyOf(values, size + roaringArray.size);
System.arraycopy(roaringArray.keys, 0, keys, size, roaringArray.size);
System.arraycopy(roaringArray.values, 0, values, size, roaringArray.size);
size += roaringArray.size;
} else if (size == 0 && roaringArray.size != 0) {
keys = Arrays.copyOf(roaringArray.keys, roaringArray.keys.length);
values = Arrays.copyOf(roaringArray.values, roaringArray.values.length);
size = roaringArray.size;
}
}
/**
* Append copies of the values AFTER a specified key (may or may not be present) to end.
*
* @param sa other array
* @param beforeStart given key is the largest key that we won't copy
*/
void appendCopiesAfter(RoaringArray sa, char beforeStart) {
int startLocation = sa.getIndex(beforeStart);
if (startLocation >= 0) {
startLocation++;
} else {
startLocation = -startLocation - 1;
}
extendArray(sa.size - startLocation);
for (int i = startLocation; i < sa.size; ++i) {
this.keys[this.size] = sa.keys[i];
this.values[this.size] = sa.values[i].clone();
this.size++;
}
}
/**
* Append copies of the values from another array, from the start
*
* @param sourceArray The array to copy from
* @param stoppingKey any equal or larger key in other array will terminate copying
*/
void appendCopiesUntil(RoaringArray sourceArray, char stoppingKey) {
for (int i = 0; i < sourceArray.size; ++i) {
if (sourceArray.keys[i] >= stoppingKey) {
break;
}
extendArray(1);
this.keys[this.size] = sourceArray.keys[i];
this.values[this.size] = sourceArray.values[i].clone();
this.size++;
}
}
/**
* Append copy of the one value from another array
*
* @param sa other array
* @param index index in the other array
*/
void appendCopy(RoaringArray sa, int index) {
extendArray(1);
this.keys[this.size] = sa.keys[index];
this.values[this.size] = sa.values[index].clone();
this.size++;
}
/**
* Append copies of the values from another array
*
* @param sa other array
* @param startingIndex starting index in the other array
* @param end endingIndex (exclusive) in the other array
*/
void appendCopy(RoaringArray sa, int startingIndex, int end) {
extendArray(end - startingIndex);
for (int i = startingIndex; i < end; ++i) {
this.keys[this.size] = sa.keys[i];
this.values[this.size] = sa.values[i].clone();
this.size++;
}
}
/**
* Append the values from another array, no copy is made (use with care)
*
* @param sa other array
* @param startingIndex starting index in the other array
* @param end endingIndex (exclusive) in the other array
*/
protected void append(RoaringArray sa, int startingIndex, int end) {
extendArray(end - startingIndex);
for (int i = startingIndex; i < end; ++i) {
this.keys[this.size] = sa.keys[i];
this.values[this.size] = sa.values[i];
this.size++;
}
}
private int binarySearch(int begin, int end, char key) {
return Util.unsignedBinarySearch(keys, begin, end, key);
}
protected void clear() {
this.keys = null;
this.values = null;
this.size = 0;
}
/**
* If possible, recover wasted memory.
*/
public void trim() {
keys = Arrays.copyOf(keys, size);
values = Arrays.copyOf(values, size);
for (Container c : values) {
c.trim();
}
}
@Override
public RoaringArray clone() throws CloneNotSupportedException {
RoaringArray sa;
sa = (RoaringArray) super.clone();
sa.keys = Arrays.copyOf(this.keys, this.size);
sa.values = Arrays.copyOf(this.values, this.size);
for (int k = 0; k < this.size; ++k) {
sa.values[k] = sa.values[k].clone();
}
sa.size = this.size;
return sa;
}
void copyRange(int begin, int end, int newBegin) {
// assuming begin <= end and newBegin < begin
final int range = end - begin;
System.arraycopy(this.keys, begin, this.keys, newBegin, range);
System.arraycopy(this.values, begin, this.values, newBegin, range);
}
/**
* Deserialize. If the DataInput is available as a byte[] or a ByteBuffer, you could prefer
* relying on {@link #deserialize(ByteBuffer)}. If the InputStream is >= 8kB, you could prefer
* relying on {@link #deserialize(DataInput, byte[])};
*
* @param in the DataInput stream
* @throws IOException Signals that an I/O exception has occurred.
* @throws InvalidRoaringFormat if a Roaring Bitmap cookie is missing.
*/
public void deserialize(DataInput in) throws IOException {
this.clear();
// little endian
final int cookie = Integer.reverseBytes(in.readInt());
if ((cookie & 0xFFFF) != SERIAL_COOKIE && cookie != SERIAL_COOKIE_NO_RUNCONTAINER) {
throw new InvalidRoaringFormat("I failed to find a valid cookie.");
}
this.size = ((cookie & 0xFFFF) == SERIAL_COOKIE) ? (cookie >>> 16) + 1
: Integer.reverseBytes(in.readInt());
// logically we cannot have more than (1<<16) containers.
if(this.size > (1<<16)) {
throw new InvalidRoaringFormat("Size too large");
}
if ((this.keys == null) || (this.keys.length < this.size)) {
this.keys = new char[this.size];
this.values = new Container[this.size];
}
byte[] bitmapOfRunContainers = null;
boolean hasrun = (cookie & 0xFFFF) == SERIAL_COOKIE;
if (hasrun) {
bitmapOfRunContainers = new byte[(size + 7) / 8];
in.readFully(bitmapOfRunContainers);
}
final char[] keys = new char[this.size];
final int[] cardinalities = new int[this.size];
final boolean[] isBitmap = new boolean[this.size];
for (int k = 0; k < this.size; ++k) {
keys[k] = Character.reverseBytes(in.readChar());
cardinalities[k] = 1 + (0xFFFF & Character.reverseBytes(in.readChar()));
isBitmap[k] = cardinalities[k] > ArrayContainer.DEFAULT_MAX_SIZE;
if (bitmapOfRunContainers != null && (bitmapOfRunContainers[k / 8] & (1 << (k % 8))) != 0) {
isBitmap[k] = false;
}
}
if ((!hasrun) || (this.size >= NO_OFFSET_THRESHOLD)) {
// skipping the offsets
in.skipBytes(this.size * 4);
}
// Reading the containers
for (int k = 0; k < this.size; ++k) {
Container val;
if (isBitmap[k]) {
final long[] bitmapArray = new long[BitmapContainer.MAX_CAPACITY / 64];
// little endian
for (int l = 0; l < bitmapArray.length; ++l) {
bitmapArray[l] = Long.reverseBytes(in.readLong());
}
val = new BitmapContainer(bitmapArray, cardinalities[k]);
} else if (bitmapOfRunContainers != null
&& ((bitmapOfRunContainers[k / 8] & (1 << (k % 8))) != 0)) {
// cf RunContainer.writeArray()
int nbrruns = (Character.reverseBytes(in.readChar()));
final char[] lengthsAndValues = new char[2 * nbrruns];
for (int j = 0; j < 2 * nbrruns; ++j) {
lengthsAndValues[j] = Character.reverseBytes(in.readChar());
}
val = new RunContainer(lengthsAndValues, nbrruns);
} else {
final char[] charArray = new char[cardinalities[k]];
for (int l = 0; l < charArray.length; ++l) {
charArray[l] = Character.reverseBytes(in.readChar());
}
val = new ArrayContainer(charArray);
}
this.keys[k] = keys[k];
this.values[k] = val;
}
}
/**
* Deserialize.
*
* @param in the DataInput stream
* @param buffer The buffer gets overwritten with data during deserialization. You can pass a NULL
* reference as a buffer. A buffer containing at least 8192 bytes might be ideal for
* performance. It is recommended to reuse the buffer between calls to deserialize (in a
* single-threaded context) for best performance.
* @throws IOException Signals that an I/O exception has occurred.
* @throws InvalidRoaringFormat if a Roaring Bitmap cookie is missing.
*/
public void deserialize(DataInput in, byte[] buffer) throws IOException {
if (buffer != null && buffer.length == 0) {
// Get rid of this useless buffer
buffer = null;
} else if (buffer != null && buffer.length % 8 != 0) {
// This is necessary not to handle manually the gap between a ShortBuffer|LongBuffer and the
// provided byte[]
throw new IllegalArgumentException(
"We need a buffer with a length multiple of 8. was length=" + buffer.length);
}
this.clear();
// little endian
final int cookie = Integer.reverseBytes(in.readInt());
if ((cookie & 0xFFFF) != SERIAL_COOKIE && cookie != SERIAL_COOKIE_NO_RUNCONTAINER) {
throw new InvalidRoaringFormat("I failed to find a valid cookie.");
}
this.size = ((cookie & 0xFFFF) == SERIAL_COOKIE) ? (cookie >>> 16) + 1
: Integer.reverseBytes(in.readInt());
// logically we cannot have more than (1<<16) containers.
if(this.size > (1<<16)) {
throw new InvalidRoaringFormat("Size too large");
}
if ((this.keys == null) || (this.keys.length < this.size)) {
this.keys = new char[this.size];
this.values = new Container[this.size];
}
byte[] bitmapOfRunContainers = null;
boolean hasrun = (cookie & 0xFFFF) == SERIAL_COOKIE;
if (hasrun) {
bitmapOfRunContainers = new byte[(size + 7) / 8];
in.readFully(bitmapOfRunContainers);
}
final char[] keys = new char[this.size];
final int[] cardinalities = new int[this.size];
final boolean[] isBitmap = new boolean[this.size];
for (int k = 0; k < this.size; ++k) {
keys[k] = Character.reverseBytes(in.readChar());
cardinalities[k] = 1 + (0xFFFF & Character.reverseBytes(in.readChar()));
isBitmap[k] = cardinalities[k] > ArrayContainer.DEFAULT_MAX_SIZE;
if (bitmapOfRunContainers != null && (bitmapOfRunContainers[k / 8] & (1 << (k % 8))) != 0) {
isBitmap[k] = false;
}
}
if ((!hasrun) || (this.size >= NO_OFFSET_THRESHOLD)) {
// skipping the offsets
in.skipBytes(this.size * 4);
}
// Reading the containers
for (int k = 0; k < this.size; ++k) {
Container val;
if (isBitmap[k]) {
final long[] bitmapArray = new long[BitmapContainer.MAX_CAPACITY / 64];
if (buffer == null) {
// a buffer to load a Container in a single .readFully
// We initialize it with the length of a BitmapContainer
buffer = new byte[(BitmapContainer.MAX_CAPACITY / 64) * 8];
}
if (buffer.length < (BitmapContainer.MAX_CAPACITY / 64) * 8) {
// We have been provided a rather small buffer
for (int iBlock = 0 ; iBlock <= 8 * bitmapArray.length / buffer.length ; iBlock++) {
int start = buffer.length * iBlock;
int end = Math.min(buffer.length * (iBlock +1 ), 8 * bitmapArray.length);
in.readFully(buffer, 0, end - start);
// little endian
ByteBuffer asByteBuffer = ByteBuffer.wrap(buffer);
asByteBuffer.order(LITTLE_ENDIAN);
LongBuffer asLongBuffer = asByteBuffer.asLongBuffer();
asLongBuffer.rewind();
asLongBuffer.get(bitmapArray, start / 8, (end - start) / 8);
}
} else {
// Read the whole bitmapContainer in a single pass
in.readFully(buffer, 0, bitmapArray.length * 8);
// little endian
ByteBuffer asByteBuffer = ByteBuffer.wrap(buffer);
asByteBuffer.order(LITTLE_ENDIAN);
LongBuffer asLongBuffer = asByteBuffer.asLongBuffer();
asLongBuffer.rewind();
asLongBuffer.get(bitmapArray);
}
val = new BitmapContainer(bitmapArray, cardinalities[k]);
} else if (bitmapOfRunContainers != null
&& ((bitmapOfRunContainers[k / 8] & (1 << (k % 8))) != 0)) {
// cf RunContainer.writeArray()
int nbrruns = (Character.reverseBytes(in.readChar()));
final char[] lengthsAndValues = new char[2 * nbrruns];
if (buffer == null && lengthsAndValues.length > (BitmapContainer.MAX_CAPACITY / 64) * 8) {
// a buffer to load a Container in a single .readFully
// We initialize it with the length of a BitmapContainer
buffer = new byte[(BitmapContainer.MAX_CAPACITY / 64) * 8];
}
if (buffer == null) {
// The RunContainer is small: skip the buffer allocation
for (int j = 0; j < lengthsAndValues.length; ++j) {
lengthsAndValues[j] = Character.reverseBytes(in.readChar());
}
} else {
for (int iBlock = 0 ; iBlock <= 2 * lengthsAndValues.length / buffer.length ; iBlock++) {
int start = buffer.length * iBlock;
int end = Math.min(buffer.length * (iBlock +1 ), 2 * lengthsAndValues.length);
in.readFully(buffer, 0, end - start);
// little endian
ByteBuffer asByteBuffer = ByteBuffer.wrap(buffer);
asByteBuffer.order(LITTLE_ENDIAN);
CharBuffer asCharBuffer = asByteBuffer.asCharBuffer();
asCharBuffer.rewind();
asCharBuffer.get(lengthsAndValues, start / 2, (end - start) / 2);
}
}
val = new RunContainer(lengthsAndValues, nbrruns);
} else {
final char[] charArray = new char[cardinalities[k]];
if (buffer == null && charArray.length > (BitmapContainer.MAX_CAPACITY / 64) * 8) {
// a buffer to load a Container in a single .readFully
// We initialize it with the length of a BitmapContainer
buffer = new byte[(BitmapContainer.MAX_CAPACITY / 64) * 8];
}
if (buffer == null) {
// The ArrayContainer is small: skip the buffer allocation
for (int j = 0; j < charArray.length; ++j) {
charArray[j] = Character.reverseBytes(in.readChar());
}
} else {
for (int iBlock = 0 ; iBlock <= 2 * charArray.length / buffer.length ; iBlock++) {
int start = buffer.length * iBlock;
int end = Math.min(buffer.length * (iBlock +1 ) , 2 * charArray.length);
in.readFully(buffer, 0, end - start);
// little endian
ByteBuffer asByteBuffer = ByteBuffer.wrap(buffer);
asByteBuffer.order(LITTLE_ENDIAN);
CharBuffer asCharBuffer = asByteBuffer.asCharBuffer();
asCharBuffer.rewind();
asCharBuffer.get(charArray, start / 2, (end - start) / 2);
}
}
val = new ArrayContainer(charArray);
}
this.keys[k] = keys[k];
this.values[k] = val;
}
}
/**
* Deserialize (retrieve) this bitmap. See format specification at
* https://github.com/RoaringBitmap/RoaringFormatSpec
*
* The current bitmap is overwritten.
*
* It is not necessary that limit() on the input ByteBuffer indicates the end of the serialized
* data.
*
* After loading this RoaringBitmap, you can advance to the rest of the data (if there
* is more) by setting bbf.position(bbf.position() + bitmap.serializedSizeInBytes());
*
* Note that the input ByteBuffer is effectively copied (with the slice operation) so you should
* expect the provided ByteBuffer position/mark/limit/order to remain unchanged.
*
* @param bbf the byte buffer (can be mapped, direct, array backed etc.
*/
public void deserialize(ByteBuffer bbf) {
this.clear();
// slice not to mutate the input ByteBuffer
ByteBuffer buffer = bbf.slice();
buffer.order(LITTLE_ENDIAN);
final int cookie = buffer.getInt();
if ((cookie & 0xFFFF) != SERIAL_COOKIE && cookie != SERIAL_COOKIE_NO_RUNCONTAINER) {
throw new InvalidRoaringFormat("I failed to find one of the right cookies. " + cookie);
}
boolean hasRunContainers = (cookie & 0xFFFF) == SERIAL_COOKIE;
this.size = hasRunContainers ? (cookie >>> 16) + 1 : buffer.getInt();
// TODO For now, we consider the limit is already set by the caller
// int theLimit = size > 0 ? computeSerializedSizeInBytes() : headerSize(hasRunContainers);
// buffer.limit(theLimit);
// logically we cannot have more than (1<<16) containers.
if(this.size > (1<<16)) {
throw new InvalidRoaringFormat("Size too large");
}
if ((this.keys == null) || (this.keys.length < this.size)) {
this.keys = new char[this.size];
this.values = new Container[this.size];
}
byte[] bitmapOfRunContainers = null;
boolean hasrun = (cookie & 0xFFFF) == SERIAL_COOKIE;
if (hasrun) {
bitmapOfRunContainers = new byte[(size + 7) / 8];
buffer.get(bitmapOfRunContainers);
}
final char[] keys = new char[this.size];
final int[] cardinalities = new int[this.size];
final boolean[] isBitmap = new boolean[this.size];
for (int k = 0; k < this.size; ++k) {
keys[k] = buffer.getChar();
cardinalities[k] = 1 + (0xFFFF & buffer.getChar());
isBitmap[k] = cardinalities[k] > ArrayContainer.DEFAULT_MAX_SIZE;
if (bitmapOfRunContainers != null && (bitmapOfRunContainers[k / 8] & (1 << (k % 8))) != 0) {
isBitmap[k] = false;
}
}
if ((!hasrun) || (this.size >= NO_OFFSET_THRESHOLD)) {
// skipping the offsets
buffer.position(buffer.position() + this.size * 4);
}
// Reading the containers
for (int k = 0; k < this.size; ++k) {
Container val;
if (isBitmap[k]) {
final long[] bitmapArray = new long[BitmapContainer.MAX_CAPACITY / 64];
buffer.asLongBuffer().get(bitmapArray);
buffer.position(buffer.position() + bitmapArray.length * 8);
val = new BitmapContainer(bitmapArray, cardinalities[k]);
} else if (bitmapOfRunContainers != null
&& ((bitmapOfRunContainers[k / 8] & (1 << (k % 8))) != 0)) {
// cf RunContainer.writeArray()
int nbrruns = (buffer.getChar());
final char[] lengthsAndValues = new char[2 * nbrruns];
buffer.asCharBuffer().get(lengthsAndValues);
buffer.position(buffer.position() + lengthsAndValues.length * 2);
val = new RunContainer(lengthsAndValues, nbrruns);
} else {
final char[] charArray = new char[cardinalities[k]];
buffer.asCharBuffer().get(charArray);
buffer.position(buffer.position() + charArray.length * 2);
val = new ArrayContainer(charArray);
}
this.keys[k] = keys[k];
this.values[k] = val;
}
}
@Override
public boolean equals(Object o) {
if (o instanceof RoaringArray) {
RoaringArray srb = (RoaringArray) o;
if (srb.size != this.size) {
return false;
}
if (ArraysShim.equals(keys, 0, size, srb.keys, 0, srb.size)) {
for (int i = 0; i < size; ++i) {
if (!values[i].equals(srb.values[i])) {
return false;
}
}
return true;
}
}
return false;
}
// make sure there is capacity for at least k more elements
void extendArray(int k) {
// size + 1 could overflow
if (this.size + k > this.keys.length) {
int newCapacity;
if (this.keys.length < 1024) {
newCapacity = 2 * (this.size + k);
} else {
newCapacity = 5 * (this.size + k) / 4;
}
this.keys = Arrays.copyOf(this.keys, newCapacity);
this.values = Arrays.copyOf(this.values, newCapacity);
}
}
// involves a binary search
protected Container getContainer(char x) {
int i = getContainerIndex(x);
if (i < 0) {
return null;
}
return this.values[i];
}
protected int getContainerIndex(char x) {
int i = this.binarySearch(0, size, x);
return i;
}
protected Container getContainerAtIndex(int i) {
return this.values[i];
}
/**
* Create a ContainerPointer for this RoaringArray
* @return a ContainerPointer
*/
public ContainerPointer getContainerPointer() {
return getContainerPointer(0);
}
/**
* Create a ContainerPointer for this RoaringArray
* @param startIndex starting index in the container list
* @return a ContainerPointer
*/
public ContainerPointer getContainerPointer(final int startIndex) {
return new ContainerPointer() {
int k = startIndex;
@Override
public void advance() {
++k;
}
@Override
public ContainerPointer clone() {
try {
return (ContainerPointer) super.clone();
} catch (CloneNotSupportedException e) {
return null;// will not happen
}
}
@Override
public int compareTo(ContainerPointer o) {
if (key() != o.key()) {
return key() - o.key();
}
return o.getCardinality() - getCardinality();
}
@Override
public int getCardinality() {
return getContainer().getCardinality();
}
@Override
public Container getContainer() {
if (k >= RoaringArray.this.size) {
return null;
}
return RoaringArray.this.values[k];
}
@Override
public boolean isBitmapContainer() {
return getContainer() instanceof BitmapContainer;
}
@Override
public boolean isRunContainer() {
return getContainer() instanceof RunContainer;
}
@Override
public char key() {
return RoaringArray.this.keys[k];
}
};
}
// involves a binary search
int getIndex(char x) {
// before the binary search, we optimize for frequent cases
if ((size == 0) || (keys[size - 1] == x)) {
return size - 1;
}
// no luck we have to go through the list
return this.binarySearch(0, size, x);
}
protected char getKeyAtIndex(int i) {
return this.keys[i];
}
@Override
public int hashCode() {
int hashvalue = 0;
for (int k = 0; k < this.size; ++k) {
hashvalue = 31 * hashvalue + keys[k] * 0xF0F0F0 + values[k].hashCode();
}
return hashvalue;
}
private boolean hasRunContainer() {
for (int k = 0; k < size; ++k) {
Container ck = values[k];
if (ck instanceof RunContainer) {
return true;
}
}
return false;
}
private int headerSize() {
if (hasRunContainer()) {
if (size < NO_OFFSET_THRESHOLD) {// for small bitmaps, we omit the offsets
return 4 + (size + 7) / 8 + 4 * size;
}
return 4 + (size + 7) / 8 + 8 * size;// - 4 because we pack the size with the cookie
} else {
return 4 + 4 + 8 * size;
}
}
// insert a new key, it is assumed that it does not exist
void insertNewKeyValueAt(int i, char key, Container value) {
extendArray(1);
System.arraycopy(keys, i, keys, i + 1, size - i);
keys[i] = key;
System.arraycopy(values, i, values, i + 1, size - i);
values[i] = value;
size++;
}
@Override
public void readExternal(ObjectInput in) throws IOException {
deserialize(in);
}
void removeAtIndex(int i) {
System.arraycopy(keys, i + 1, keys, i, size - i - 1);
keys[size - 1] = 0;
System.arraycopy(values, i + 1, values, i, size - i - 1);
values[size - 1] = null;
size--;
}
void removeIndexRange(int begin, int end) {
if (end <= begin) {
return;
}
final int range = end - begin;
System.arraycopy(keys, end, keys, begin, size - end);
System.arraycopy(values, end, values, begin, size - end);
for (int i = 1; i <= range; ++i) {
keys[size - i] = 0;
values[size - i] = null;
}
size -= range;
}
void replaceKeyAndContainerAtIndex(int i, char key, Container c) {
this.keys[i] = key;
this.values[i] = c;
}
void resize(int newLength) {
Arrays.fill(this.keys, newLength, this.size, (char) 0);
Arrays.fill(this.values, newLength, this.size, null);
this.size = newLength;
}
/**
* Serialize.
*
* The current bitmap is not modified.
*
* @param out the DataOutput stream
* @throws IOException Signals that an I/O exception has occurred.
*/
public void serialize(DataOutput out) throws IOException {
int startOffset = 0;
boolean hasrun = hasRunContainer();
if (hasrun) {
out.writeInt(Integer.reverseBytes(SERIAL_COOKIE | ((size - 1) << 16)));
byte[] bitmapOfRunContainers = new byte[(size + 7) / 8];
for (int i = 0; i < size; ++i) {
if (this.values[i] instanceof RunContainer) {
bitmapOfRunContainers[i / 8] |= (1 << (i % 8));
}
}
out.write(bitmapOfRunContainers);
if (this.size < NO_OFFSET_THRESHOLD) {
startOffset = 4 + 4 * this.size + bitmapOfRunContainers.length;
} else {
startOffset = 4 + 8 * this.size + bitmapOfRunContainers.length;
}
} else { // backwards compatibility
out.writeInt(Integer.reverseBytes(SERIAL_COOKIE_NO_RUNCONTAINER));
out.writeInt(Integer.reverseBytes(size));
startOffset = 4 + 4 + 4 * this.size + 4 * this.size;
}
for (int k = 0; k < size; ++k) {
out.writeShort(Character.reverseBytes(this.keys[k]));
out.writeShort(Character.reverseBytes((char) (this.values[k].getCardinality() - 1)));
}
if ((!hasrun) || (this.size >= NO_OFFSET_THRESHOLD)) {
// writing the containers offsets
for (int k = 0; k < this.size; k++) {
out.writeInt(Integer.reverseBytes(startOffset));
startOffset = startOffset + this.values[k].getArraySizeInBytes();
}
}
for (int k = 0; k < size; ++k) {
values[k].writeArray(out);
}
}
/**
* Serialize.
*
* The current bitmap is not modified.
*
* @param buffer the ByteBuffer to write to
*/
public void serialize(ByteBuffer buffer) {
ByteBuffer buf = buffer.order() == LITTLE_ENDIAN ? buffer : buffer.slice().order(LITTLE_ENDIAN);
int startOffset;
boolean hasrun = hasRunContainer();
if (hasrun) {
buf.putInt(SERIAL_COOKIE | ((size - 1) << 16));
int offset = buf.position();
for (int i = 0; i < size; i += 8) {
int runMarker = 0;
for (int j = 0; j < 8 && i + j < size; ++j) {
if (values[i + j] instanceof RunContainer) {
runMarker |= (1 << j);
}
}
buf.put((byte)runMarker);
}
int runMarkersLength = buf.position() - offset;
if (this.size < NO_OFFSET_THRESHOLD) {
startOffset = 4 + 4 * this.size + runMarkersLength;
} else {
startOffset = 4 + 8 * this.size + runMarkersLength;
}
} else { // backwards compatibility
buf.putInt(SERIAL_COOKIE_NO_RUNCONTAINER);
buf.putInt(size);
startOffset = 4 + 4 + 4 * this.size + 4 * this.size;
}
for (int k = 0; k < size; ++k) {
buf.putChar(this.keys[k]);
buf.putChar((char) (this.values[k].getCardinality() - 1));
}
if ((!hasrun) || (this.size >= NO_OFFSET_THRESHOLD)) {
// writing the containers offsets
for (int k = 0; k < this.size; ++k) {
buf.putInt(startOffset);
startOffset = startOffset + this.values[k].getArraySizeInBytes();
}
}
for (int k = 0; k < size; ++k) {
values[k].writeArray(buf);
}
if (buf != buffer) {
buffer.position(buffer.position() + buf.position());
}
}
/**
* Report the number of bytes required for serialization.
*
* @return the size in bytes
*/
public int serializedSizeInBytes() {
int count = headerSize();
for (int k = 0; k < size; ++k) {
count += values[k].getArraySizeInBytes();
}
return count;
}
void setContainerAtIndex(int i, Container c) {
this.values[i] = c;
}
protected int size() {
return this.size;
}
@Override
public void writeExternal(ObjectOutput out) throws IOException {
serialize(out);
}
/**
* Gets the first value in the array
* @return the first value in the array
* @throws NoSuchElementException if empty
*/
public int first() {
assertNonEmpty();
char firstKey = keys[0];
Container container = values[0];
return firstKey << 16 | container.first();
}
/**
* Gets the last value in the array
* @return the last value in the array
* @throws NoSuchElementException if empty
*/
public int last() {
assertNonEmpty();
char lastKey = keys[size - 1];
Container container = values[size - 1];
return lastKey << 16 | container.last();
}
private void assertNonEmpty() {
if(size == 0) {
throw new NoSuchElementException("Empty RoaringArray");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy