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

org.apache.pulsar.common.util.collections.LongPairRangeSet Maven / Gradle / Ivy

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 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 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