org.elasticsearch.compute.operator.mvdedupe.MultivalueDedupeLong Maven / Gradle / Ivy
/*
* Copyright Elasticsearch B.V. and/or licensed to Elasticsearch B.V. under one
* or more contributor license agreements. Licensed under the Elastic License
* 2.0; you may not use this file except in compliance with the Elastic License
* 2.0.
*/
package org.elasticsearch.compute.operator.mvdedupe;
import org.apache.lucene.util.ArrayUtil;
import org.elasticsearch.common.util.LongHash;
import org.elasticsearch.compute.aggregation.GroupingAggregatorFunction;
import org.elasticsearch.compute.aggregation.blockhash.BlockHash;
import org.elasticsearch.compute.data.Block;
import org.elasticsearch.compute.data.BlockFactory;
import org.elasticsearch.compute.data.IntBlock;
import org.elasticsearch.compute.data.LongBlock;
import java.util.Arrays;
/**
* Removes duplicate values from multivalued positions.
* This class is generated. Edit {@code X-MultivalueDedupe.java.st} instead.
*/
public class MultivalueDedupeLong {
/**
* The number of entries before we switch from and {@code n^2} strategy
* with low overhead to an {@code n*log(n)} strategy with higher overhead.
* The choice of number has been experimentally derived.
*/
static final int ALWAYS_COPY_MISSING = 300;
/**
* The {@link Block} being deduplicated.
*/
final LongBlock block;
/**
* Oversized array of values that contains deduplicated values after
* running {@link #copyMissing} and sorted values after calling
* {@link #copyAndSort}
*/
long[] work = new long[ArrayUtil.oversize(2, Long.BYTES)];
/**
* After calling {@link #copyMissing} or {@link #copyAndSort} this is
* the number of values in {@link #work} for the current position.
*/
int w;
public MultivalueDedupeLong(LongBlock block) {
this.block = block;
}
/**
* Remove duplicate values from each position and write the results to a
* {@link Block} using an adaptive algorithm based on the size of the input list.
*/
public LongBlock dedupeToBlockAdaptive(BlockFactory blockFactory) {
if (block.mvDeduplicated()) {
block.incRef();
return block;
}
try (LongBlock.Builder builder = blockFactory.newLongBlockBuilder(block.getPositionCount())) {
for (int p = 0; p < block.getPositionCount(); p++) {
int count = block.getValueCount(p);
int first = block.getFirstValueIndex(p);
switch (count) {
case 0 -> builder.appendNull();
case 1 -> builder.appendLong(block.getLong(first));
default -> {
/*
* It's better to copyMissing when there are few unique values
* and better to copy and sort when there are many unique values.
* The more duplicate values there are the more comparatively worse
* copyAndSort is. But we don't know how many unique values there
* because our job is to find them. So we use the count of values
* as a proxy that is fast to test. It's not always going to be
* optimal but it has the nice property of being quite quick on
* short lists and not n^2 levels of terrible on long ones.
*
* It'd also be possible to make a truly hybrid mechanism that
* switches from copyMissing to copyUnique once it collects enough
* unique values. The trouble is that the switch is expensive and
* makes kind of a "hole" in the performance of that mechanism where
* you may as well have just gone with either of the two other
* strategies. So we just don't try it for now.
*/
if (count < ALWAYS_COPY_MISSING) {
copyMissing(first, count);
writeUniquedWork(builder);
} else {
copyAndSort(first, count);
deduplicatedSortedWork(builder);
}
}
}
}
return builder.build();
}
}
/**
* Remove duplicate values from each position and write the results to a
* {@link Block} using an algorithm with very low overhead but {@code n^2}
* case complexity for larger. Prefer {@link #dedupeToBlockAdaptive}
* which picks based on the number of elements at each position.
*/
public LongBlock dedupeToBlockUsingCopyAndSort(BlockFactory blockFactory) {
if (block.mvDeduplicated()) {
block.incRef();
return block;
}
try (LongBlock.Builder builder = blockFactory.newLongBlockBuilder(block.getPositionCount())) {
for (int p = 0; p < block.getPositionCount(); p++) {
int count = block.getValueCount(p);
int first = block.getFirstValueIndex(p);
switch (count) {
case 0 -> builder.appendNull();
case 1 -> builder.appendLong(block.getLong(first));
default -> {
copyAndSort(first, count);
deduplicatedSortedWork(builder);
}
}
}
return builder.build();
}
}
/**
* Remove duplicate values from each position and write the results to a
* {@link Block} using an algorithm that sorts all values. It has a higher
* overhead for small numbers of values at each position than
* {@link #dedupeToBlockUsingCopyMissing} for large numbers of values the
* performance is dominated by the {@code n*log n} sort. Prefer
* {@link #dedupeToBlockAdaptive} unless you need the results sorted.
*/
public LongBlock dedupeToBlockUsingCopyMissing(BlockFactory blockFactory) {
if (block.mvDeduplicated()) {
block.incRef();
return block;
}
try (LongBlock.Builder builder = blockFactory.newLongBlockBuilder(block.getPositionCount())) {
for (int p = 0; p < block.getPositionCount(); p++) {
int count = block.getValueCount(p);
int first = block.getFirstValueIndex(p);
switch (count) {
case 0 -> builder.appendNull();
case 1 -> builder.appendLong(block.getLong(first));
default -> {
copyMissing(first, count);
writeUniquedWork(builder);
}
}
}
return builder.build();
}
}
/**
* Sort values from each position and write the results to a {@link Block}.
*/
public LongBlock sortToBlock(BlockFactory blockFactory, boolean ascending) {
try (LongBlock.Builder builder = blockFactory.newLongBlockBuilder(block.getPositionCount())) {
for (int p = 0; p < block.getPositionCount(); p++) {
int count = block.getValueCount(p);
int first = block.getFirstValueIndex(p);
switch (count) {
case 0 -> builder.appendNull();
case 1 -> builder.appendLong(block.getLong(first));
default -> {
copyAndSort(first, count);
writeSortedWork(builder, ascending);
}
}
}
return builder.build();
}
}
/**
* Dedupe values, add them to the hash, and build an {@link IntBlock} of
* their hashes. This block is suitable for passing as the grouping block
* to a {@link GroupingAggregatorFunction}.
*/
public MultivalueDedupe.HashResult hashAdd(BlockFactory blockFactory, LongHash hash) {
try (IntBlock.Builder builder = blockFactory.newIntBlockBuilder(block.getPositionCount())) {
boolean sawNull = false;
for (int p = 0; p < block.getPositionCount(); p++) {
int count = block.getValueCount(p);
int first = block.getFirstValueIndex(p);
switch (count) {
case 0 -> {
sawNull = true;
builder.appendInt(0);
}
case 1 -> {
long v = block.getLong(first);
hashAdd(builder, hash, v);
}
default -> {
if (count < ALWAYS_COPY_MISSING) {
copyMissing(first, count);
hashAddUniquedWork(hash, builder);
} else {
copyAndSort(first, count);
hashAddSortedWork(hash, builder);
}
}
}
}
return new MultivalueDedupe.HashResult(builder.build(), sawNull);
}
}
/**
* Dedupe values and build an {@link IntBlock} of their hashes. This block is
* suitable for passing as the grouping block to a {@link GroupingAggregatorFunction}.
*/
public IntBlock hashLookup(BlockFactory blockFactory, LongHash hash) {
try (IntBlock.Builder builder = blockFactory.newIntBlockBuilder(block.getPositionCount())) {
for (int p = 0; p < block.getPositionCount(); p++) {
int count = block.getValueCount(p);
int first = block.getFirstValueIndex(p);
switch (count) {
case 0 -> builder.appendInt(0);
case 1 -> {
long v = block.getLong(first);
hashLookupSingle(builder, hash, v);
}
default -> {
if (count < ALWAYS_COPY_MISSING) {
copyMissing(first, count);
hashLookupUniquedWork(hash, builder);
} else {
copyAndSort(first, count);
hashLookupSortedWork(hash, builder);
}
}
}
}
return builder.build();
}
}
/**
* Build a {@link BatchEncoder} which deduplicates values at each position
* and then encodes the results into a {@link byte[]} which can be used for
* things like hashing many fields together.
*/
public BatchEncoder batchEncoder(int batchSize) {
block.incRef();
return new BatchEncoder.Longs(batchSize) {
@Override
protected void readNextBatch() {
int position = firstPosition();
if (w > 0) {
// The last block didn't fit so we have to *make* it fit
ensureCapacity(w);
startPosition();
encodeUniquedWork(this);
endPosition();
position++;
}
for (; position < block.getPositionCount(); position++) {
int count = block.getValueCount(position);
int first = block.getFirstValueIndex(position);
switch (count) {
case 0 -> encodeNull();
case 1 -> {
long v = block.getLong(first);
if (hasCapacity(1)) {
startPosition();
encode(v);
endPosition();
} else {
work[0] = v;
w = 1;
return;
}
}
default -> {
if (count < ALWAYS_COPY_MISSING) {
copyMissing(first, count);
} else {
copyAndSort(first, count);
convertSortedWorkToUnique();
}
if (hasCapacity(w)) {
startPosition();
encodeUniquedWork(this);
endPosition();
} else {
return;
}
}
}
}
}
@Override
public void close() {
block.decRef();
}
};
}
/**
* Copy all value from the position into {@link #work} and then
* sorts it {@code n * log(n)}.
*/
void copyAndSort(int first, int count) {
grow(count);
int end = first + count;
w = 0;
for (int i = first; i < end; i++) {
work[w++] = block.getLong(i);
}
Arrays.sort(work, 0, w);
}
/**
* Fill {@link #work} with the unique values in the position by scanning
* all fields already copied {@code n^2}.
*/
void copyMissing(int first, int count) {
grow(count);
int end = first + count;
work[0] = block.getLong(first);
w = 1;
i: for (int i = first + 1; i < end; i++) {
long v = block.getLong(i);
for (int j = 0; j < w; j++) {
if (v == work[j]) {
continue i;
}
}
work[w++] = v;
}
}
/**
* Writes an already deduplicated {@link #work} to a {@link LongBlock.Builder}.
*/
private void writeUniquedWork(LongBlock.Builder builder) {
if (w == 1) {
builder.appendLong(work[0]);
return;
}
builder.beginPositionEntry();
for (int i = 0; i < w; i++) {
builder.appendLong(work[i]);
}
builder.endPositionEntry();
}
/**
* Writes a sorted {@link #work} to a {@link LongBlock.Builder}, skipping duplicates.
*/
private void deduplicatedSortedWork(LongBlock.Builder builder) {
builder.beginPositionEntry();
long prev = work[0];
builder.appendLong(prev);
for (int i = 1; i < w; i++) {
if (prev != work[i]) {
prev = work[i];
builder.appendLong(prev);
}
}
builder.endPositionEntry();
}
/**
* Writes a {@link #work} to a {@link LongBlock.Builder}.
*/
private void writeSortedWork(LongBlock.Builder builder, boolean ascending) {
builder.beginPositionEntry();
for (int i = 0; i < w; i++) {
if (ascending) {
builder.appendLong(work[i]);
} else {
builder.appendLong(work[w - i - 1]);
}
}
builder.endPositionEntry();
}
/**
* Writes an already deduplicated {@link #work} to a hash.
*/
private void hashAddUniquedWork(LongHash hash, IntBlock.Builder builder) {
if (w == 1) {
hashAdd(builder, hash, work[0]);
return;
}
builder.beginPositionEntry();
for (int i = 0; i < w; i++) {
hashAdd(builder, hash, work[i]);
}
builder.endPositionEntry();
}
/**
* Writes a sorted {@link #work} to a hash, skipping duplicates.
*/
private void hashAddSortedWork(LongHash hash, IntBlock.Builder builder) {
if (w == 1) {
hashAdd(builder, hash, work[0]);
return;
}
builder.beginPositionEntry();
long prev = work[0];
hashAdd(builder, hash, prev);
for (int i = 1; i < w; i++) {
if (false == valuesEqual(prev, work[i])) {
prev = work[i];
hashAdd(builder, hash, prev);
}
}
builder.endPositionEntry();
}
/**
* Looks up an already deduplicated {@link #work} to a hash.
*/
private void hashLookupUniquedWork(LongHash hash, IntBlock.Builder builder) {
if (w == 1) {
hashLookupSingle(builder, hash, work[0]);
return;
}
int i = 1;
long firstLookup = hashLookup(hash, work[0]);
while (firstLookup < 0) {
if (i >= w) {
// Didn't find any values
builder.appendNull();
return;
}
firstLookup = hashLookup(hash, work[i]);
i++;
}
/*
* Step 2 - find the next unique value in the hash
*/
boolean foundSecond = false;
while (i < w) {
long nextLookup = hashLookup(hash, work[i]);
if (nextLookup >= 0) {
builder.beginPositionEntry();
appendFound(builder, firstLookup);
appendFound(builder, nextLookup);
i++;
foundSecond = true;
break;
}
i++;
}
/*
* Step 3a - we didn't find a second value, just emit the first one
*/
if (false == foundSecond) {
appendFound(builder, firstLookup);
return;
}
/*
* Step 3b - we found a second value, search for more
*/
while (i < w) {
long nextLookup = hashLookup(hash, work[i]);
if (nextLookup >= 0) {
appendFound(builder, nextLookup);
}
i++;
}
builder.endPositionEntry();
}
/**
* Looks up a sorted {@link #work} to a hash, skipping duplicates.
*/
private void hashLookupSortedWork(LongHash hash, IntBlock.Builder builder) {
if (w == 1) {
hashLookupSingle(builder, hash, work[0]);
return;
}
/*
* Step 1 - find the first unique value in the hash
* i will contain the next value to probe
* prev will contain the first value in the array contained in the hash
* firstLookup will contain the first value in the hash
*/
int i = 1;
long prev = work[0];
long firstLookup = hashLookup(hash, prev);
while (firstLookup < 0) {
if (i >= w) {
// Didn't find any values
builder.appendNull();
return;
}
prev = work[i];
firstLookup = hashLookup(hash, prev);
i++;
}
/*
* Step 2 - find the next unique value in the hash
*/
boolean foundSecond = false;
while (i < w) {
if (false == valuesEqual(prev, work[i])) {
long nextLookup = hashLookup(hash, work[i]);
if (nextLookup >= 0) {
prev = work[i];
builder.beginPositionEntry();
appendFound(builder, firstLookup);
appendFound(builder, nextLookup);
i++;
foundSecond = true;
break;
}
}
i++;
}
/*
* Step 3a - we didn't find a second value, just emit the first one
*/
if (false == foundSecond) {
appendFound(builder, firstLookup);
return;
}
/*
* Step 3b - we found a second value, search for more
*/
while (i < w) {
if (false == valuesEqual(prev, work[i])) {
long nextLookup = hashLookup(hash, work[i]);
if (nextLookup >= 0) {
prev = work[i];
appendFound(builder, nextLookup);
}
}
i++;
}
builder.endPositionEntry();
}
/**
* Writes a deduplicated {@link #work} to a {@link BatchEncoder.Longs}.
*/
private void encodeUniquedWork(BatchEncoder.Longs encoder) {
for (int i = 0; i < w; i++) {
encoder.encode(work[i]);
}
}
/**
* Converts {@link #work} from sorted array to a deduplicated array.
*/
private void convertSortedWorkToUnique() {
long prev = work[0];
int end = w;
w = 1;
for (int i = 1; i < end; i++) {
if (false == valuesEqual(prev, work[i])) {
prev = work[i];
work[w++] = prev;
}
}
}
private void grow(int size) {
work = ArrayUtil.grow(work, size);
}
private void hashAdd(IntBlock.Builder builder, LongHash hash, long v) {
appendFound(builder, hash.add(v));
}
private long hashLookup(LongHash hash, long v) {
return hash.find(v);
}
private void hashLookupSingle(IntBlock.Builder builder, LongHash hash, long v) {
long found = hashLookup(hash, v);
if (found >= 0) {
appendFound(builder, found);
} else {
builder.appendNull();
}
}
private void appendFound(IntBlock.Builder builder, long found) {
builder.appendInt(Math.toIntExact(BlockHash.hashOrdToGroupNullReserved(found)));
}
private static boolean valuesEqual(long lhs, long rhs) {
return lhs == rhs;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy