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

org.roaringbitmap.ContainerAppender Maven / Gradle / Ivy

The newest version!
package org.roaringbitmap;


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 ContainerAppender is to get bitmaps quickly.
 * You should benchmark your particular use case to see if it helps.
 *
 * 
 * {@code
 *
 *     //...
 *
 *     RoaringBitmapWriter writer =
 *        RoaringBitmapWriter.writer().get();
 *     for (int i :....) {
 *       writer.add(i);
 *     }
 *     writer.flush(); // important
 * }
 * 
*/ public class ContainerAppender, T extends BitmapDataProvider & AppendableStorage> implements RoaringBitmapWriter { private final boolean doPartialSort; private final Supplier newContainer; private final Supplier newUnderlying; private C container; private T underlying; private int currentKey; /** * Initialize an ContainerAppender with a receiving bitmap * */ ContainerAppender(boolean doPartialSort, Supplier newUnderlying, Supplier newContainer) { this.doPartialSort = doPartialSort; this.newUnderlying = newUnderlying; this.underlying = newUnderlying.get(); this.newContainer = newContainer; this.container = newContainer.get(); } /** * Grab a reference to the underlying bitmap * * @return the underlying bitmap */ 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 = toIntUnsigned(highbits(value)); if (key != currentKey) { if (key < currentKey) { underlying.add(value); return; } else { appendToUnderlying(); currentKey = key; } } C tmp = container.add(lowbits(value)); if (tmp != container) { container = tmp; } } @Override public void add(long min, long max) { appendToUnderlying(); underlying.add(min, max); int mark = (int)((max >>> 16) + 1); if (currentKey < mark) { currentKey = mark; } } @Override public void addMany(int... values) { if (doPartialSort) { partialRadixSort(values); } for (int i : values) { add(i); } } /** * Ensures that any buffered additions are flushed to the underlying bitmap. */ @Override public void flush() { currentKey += appendToUnderlying(); } @Override public void reset() { currentKey = 0; container = newContainer.get(); underlying = newUnderlying.get(); } private int appendToUnderlying() { if (!container.isEmpty()) { assert currentKey <= 0xFFFF; underlying.append((short)currentKey, container.runOptimize()); container = newContainer.get(); return 1; } return 0; } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy