org.apache.pulsar.common.util.collections.LongPairRangeSet Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of pulsar-common Show documentation
Show all versions of pulsar-common Show documentation
Common libraries needed by client/broker/tools
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.pulsar.common.util.collections;
import com.google.common.collect.Lists;
import com.google.common.collect.Range;
import com.google.common.collect.RangeSet;
import com.google.common.collect.TreeRangeSet;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Set;
import lombok.EqualsAndHashCode;
/**
* A set comprising zero or more ranges type of key-value pair.
*/
public interface LongPairRangeSet> {
/**
* Adds the specified range (range that contains all values strictly greater than {@code
* lower} and less than or equal to {@code upper}.) to this {@code RangeSet} (optional operation). That is, for equal
* range sets a and b, the result of {@code a.add(range)} is that {@code a} will be the minimal range set for which
* both {@code a.enclosesAll(b)} and {@code a.encloses(range)}.
*
*
*
* @param lowerKey : value for key of lowerEndpoint of Range
* @param lowerValue: value for value of lowerEndpoint of Range
* @param upperKey : value for key of upperEndpoint of Range
* @param upperValue: value for value of upperEndpoint of Range
*
*/
void addOpenClosed(long lowerKey, long lowerValue, long upperKey, long upperValue);
/** Determines whether any of this range set's member ranges contains {@code value}. */
boolean contains(long key, long value);
/**
* Returns the unique range from this range set that {@linkplain Range#contains contains} {@code value}, or
* {@code null} if this range set does not contain {@code value}.
*/
Range rangeContaining(long key, long value);
/**
* Remove range that contains all values less than or equal to given key-value.
*
* @param key
* @param value
*/
void removeAtMost(long key, long value);
boolean isEmpty();
void clear();
/**
* Returns the minimal range which {@linkplain Range#encloses(Range) encloses} all ranges in this range set.
*
* @return
*/
Range span();
/**
* Returns a view of the {@linkplain Range#isConnected disconnected} ranges that make up this range set.
*
* @return
*/
Collection> asRanges();
/**
* Performs the given action for each entry in this map until all entries have been processed
* or action returns "false". Unless otherwise specified by the implementing class,
* actions are performed in the order of entry set iteration (if an iteration order is specified.)
*
* @param action
*/
void forEach(RangeProcessor action);
/**
* Performs the given action for each entry in this map until all entries have been processed
* or action returns "false". Unless otherwise specified by the implementing class,
* actions are performed in the order of entry set iteration (if an iteration order is specified.)
*
* @param action
* @param consumer
*/
void forEach(RangeProcessor action, LongPairConsumer extends T> consumer);
/**
* Performs the given action for each entry in this map until all entries have been processed
* or action returns "false". Unless otherwise specified by the implementing class,
* actions are performed in the order of entry set iteration (if an iteration order is specified.)
*
* This method is optimized on reducing intermediate object creation.
* {@param action} to do iteration jobs.
*
*/
void forEachRawRange(RawRangeProcessor action);
/**
* Returns total number of ranges into the set.
*
* @return
*/
int size();
/**
* It returns very first smallest range in the rangeSet.
*
* @return first smallest range into the set
*/
Range firstRange();
/**
* It returns very last biggest range in the rangeSet.
*
* @return last biggest range into the set
*/
Range lastRange();
default Map toRanges(int maxRanges) {
throw new UnsupportedOperationException();
}
/**
* Build {@link LongPairRangeSet} using internal ranges returned by {@link #toRanges(int)} .
*
* @param ranges
*/
default void build(Map ranges) {
throw new UnsupportedOperationException();
}
/**
* Return the number bit sets to true from lower (inclusive) to upper (inclusive).
*/
int cardinality(long lowerKey, long lowerValue, long upperKey, long upperValue);
/**
* Represents a function that accepts two long arguments and produces a result.
*
* @param the type of the result.
*/
interface LongPairConsumer {
T apply(long key, long value);
}
/**
* Represents a function that accepts result and produces a LongPair.
* Reverse ops of `LongPairConsumer`
* @param the type of the result.
*/
interface RangeBoundConsumer {
LongPair apply(T bound);
}
/**
* The interface exposing a method for processing of ranges.
* @param - The incoming type of data in the range object.
*/
interface RangeProcessor> {
/**
*
* @param range
* @return false if there is no further processing required
*/
boolean process(Range range);
}
/**
* The interface exposing a method for processing raw form of ranges.
* This method will omit the process to convert (long, long) to `T`
* create less object during the iteration.
* the parameter is the same as {@linkplain RangeProcessor} which
* means (lowerKey,lowerValue) in open bound
* (upperKey, upperValue) in close bound in Range
*/
interface RawRangeProcessor {
boolean processRawRange(long lowerKey, long lowerValue,
long upperKey, long upperValue);
}
/**
* This class is a simple key-value data structure.
*/
@EqualsAndHashCode
class LongPair implements Comparable {
@SuppressWarnings("checkstyle:ConstantName")
public static final LongPair earliest = new LongPair(-1, -1);
@SuppressWarnings("checkstyle:ConstantName")
public static final LongPair latest = new LongPair(Integer.MAX_VALUE, Integer.MAX_VALUE);
private long key;
private long value;
public LongPair(long key, long value) {
this.key = key;
this.value = value;
}
public long getKey() {
return this.key;
}
public long getValue() {
return this.value;
}
@Override
public int compareTo(LongPair that) {
if (this.key != that.key) {
return this.key < that.key ? -1 : 1;
}
if (this.value != that.value) {
return this.value < that.value ? -1 : 1;
}
return 0;
}
@Override
public String toString() {
return key + ":" + value;
}
}
/**
* Generic implementation of a default range set.
*
* @param the type of values in ranges.
*/
class DefaultRangeSet> implements LongPairRangeSet {
RangeSet set = TreeRangeSet.create();
private final LongPairConsumer consumer;
private final RangeBoundConsumer rangeEndPointConsumer;
public DefaultRangeSet(LongPairConsumer consumer, RangeBoundConsumer reverseConsumer) {
this.consumer = consumer;
this.rangeEndPointConsumer = reverseConsumer;
}
@Override
public void clear() {
set.clear();
}
@Override
public void addOpenClosed(long key1, long value1, long key2, long value2) {
set.add(Range.openClosed(consumer.apply(key1, value1), consumer.apply(key2, value2)));
}
public boolean contains(T position) {
return set.contains(position);
}
public Range rangeContaining(T position) {
return set.rangeContaining(position);
}
@Override
public Range rangeContaining(long key, long value) {
return this.rangeContaining(consumer.apply(key, value));
}
public void remove(Range range) {
set.remove(range);
}
@Override
public void removeAtMost(long key, long value) {
set.remove(Range.atMost(consumer.apply(key, value)));
}
@Override
public boolean isEmpty() {
return set.isEmpty();
}
@Override
public Range span() {
try {
return set.span();
} catch (NoSuchElementException e) {
return null;
}
}
@Override
public Set> asRanges() {
return set.asRanges();
}
@Override
public void forEach(RangeProcessor action) {
forEach(action, consumer);
}
@Override
public void forEach(RangeProcessor action, LongPairConsumer extends T> outerConsumer) {
for (Range range : asRanges()) {
if (!action.process(range)) {
break;
}
}
}
@Override
public void forEachRawRange(RawRangeProcessor action) {
for (Range range : asRanges()) {
LongPair lowerEndpoint = this.rangeEndPointConsumer.apply(range.lowerEndpoint());
LongPair upperEndpoint = this.rangeEndPointConsumer.apply(range.upperEndpoint());
if (!action.processRawRange(lowerEndpoint.key, lowerEndpoint.value,
upperEndpoint.key, upperEndpoint.value)) {
break;
}
}
}
@Override
public boolean contains(long key, long value) {
return this.contains(consumer.apply(key, value));
}
@Override
public Range firstRange() {
Iterator> iterable = set.asRanges().iterator();
if (iterable.hasNext()) {
return iterable.next();
}
return null;
}
@Override
public Range lastRange() {
if (set.asRanges().isEmpty()) {
return null;
}
List> list = Lists.newArrayList(set.asRanges().iterator());
return list.get(list.size() - 1);
}
@Override
public int cardinality(long lowerKey, long lowerValue, long upperKey, long upperValue) {
throw new UnsupportedOperationException();
}
@Override
public int size() {
return set.asRanges().size();
}
@Override
public String toString() {
return set.toString();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy