org.roaringbitmap.ConstantMemoryContainerAppender Maven / Gradle / Ivy
package org.roaringbitmap;
import java.util.Arrays;
import java.util.function.Supplier;
import static org.roaringbitmap.Util.*;
/**
* This class can be used to write quickly values to a bitmap.
* The values are expected to be (increasing) sorted order.
* Values are first written to a temporary internal buffer, but
* the underlying bitmap can forcefully synchronize by calling "flush"
* (although calling flush to often would defeat the performance
* purpose of this class).
* The main use case for an ConstantMemoryContainerAppender is to get bitmaps quickly.
* You should benchmark your particular use case to see if it helps.
*
*
* {@code
*
* //...
*
*
* RoaringBitmapWriter writer =
* RoaringBitmapWriter.writer().constantMemory().get();
* for (int i :....) {
* writer.add(i);
* }
* writer.flush(); // important
* }
*
*/
public class ConstantMemoryContainerAppender> implements RoaringBitmapWriter {
private final boolean doPartialSort;
private final boolean runCompress;
private static final int WORD_COUNT = 1 << 10;
private final long[] bitmap;
private final Supplier newUnderlying;
private T underlying;
private boolean dirty = false;
private int currentKey;
/**
* Initialize an ConstantMemoryContainerAppender with a receiving bitmap
*
* @param doPartialSort indicates whether to sort the upper 16 bits of input data in addMany
* @param runCompress whether to run compress appended containers
* @param newUnderlying supplier of bitmaps where the data gets written
*/
ConstantMemoryContainerAppender(boolean doPartialSort,
boolean runCompress,
Supplier newUnderlying) {
this.newUnderlying = newUnderlying;
this.underlying = newUnderlying.get();
this.doPartialSort = doPartialSort;
this.runCompress = runCompress;
this.bitmap = new long[WORD_COUNT];
}
/**
* Grab a reference to the underlying bitmap
*
* @return the underlying bitmap
*/
@Override
public T getUnderlying() {
return underlying;
}
/**
* Adds the value to the underlying bitmap. The data might
* be added to a temporary buffer. You should call "flush"
* when you are done.
*
* @param value the value to add.
*/
@Override
public void add(int value) {
int key = (highbits(value));
if (key != currentKey) {
if (key < currentKey) {
underlying.add(value);
return;
} else {
appendToUnderlying();
currentKey = key;
}
}
int low = (lowbits(value));
bitmap[(low >>> 6)] |= (1L << low);
dirty = true;
}
@Override
public void addMany(int... values) {
if (doPartialSort) {
partialRadixSort(values);
}
for (int value : values) {
add(value);
}
}
@Override
public void add(long min, long max) {
appendToUnderlying();
underlying.add(min, max);
int mark = (int)((max >>> 16) + 1);
if (currentKey < mark) {
currentKey = mark;
}
}
/**
* Ensures that any buffered additions are flushed to the underlying bitmap.
*/
@Override
public void flush() {
currentKey += appendToUnderlying();
}
@Override
public void reset() {
currentKey = 0;
underlying = newUnderlying.get();
dirty = false;
}
private Container chooseBestContainer() {
Container container = new BitmapContainer(bitmap, -1).repairAfterLazy();
if (runCompress) {
container = container.runOptimize();
}
return container instanceof BitmapContainer ? container.clone() : container;
}
private int appendToUnderlying() {
if (dirty) {
assert currentKey <= 0xFFFF;
underlying.append((char) currentKey, chooseBestContainer());
Arrays.fill(bitmap, 0L);
dirty = false;
return 1;
}
return 0;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy