FluentBitSet.java

  1. /*
  2.  * Licensed to the Apache Software Foundation (ASF) under one or more
  3.  * contributor license agreements.  See the NOTICE file distributed with
  4.  * this work for additional information regarding copyright ownership.
  5.  * The ASF licenses this file to You under the Apache License, Version 2.0
  6.  * (the "License"); you may not use this file except in compliance with
  7.  * the License.  You may obtain a copy of the License at
  8.  *
  9.  *      http://www.apache.org/licenses/LICENSE-2.0
  10.  *
  11.  * Unless required by applicable law or agreed to in writing, software
  12.  * distributed under the License is distributed on an "AS IS" BASIS,
  13.  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14.  * See the License for the specific language governing permissions and
  15.  * limitations under the License.
  16.  */
  17. package org.apache.commons.lang3.util;

  18. import java.io.Serializable;
  19. import java.util.BitSet;
  20. import java.util.Objects;
  21. import java.util.stream.IntStream;

  22. /**
  23.  * A fluent {@link BitSet} with additional operations.
  24.  * <p>
  25.  * Originally from Apache Commons VFS with more added to act as a fluent replacement for {@link java.util.BitSet}.
  26.  * </p>
  27.  * @since 3.13.0
  28.  */
  29. public final class FluentBitSet implements Cloneable, Serializable {

  30.     private static final long serialVersionUID = 1L;

  31.     /**
  32.      * Working BitSet.
  33.      */
  34.     private final BitSet bitSet;

  35.     /**
  36.      * Creates a new bit set. All bits are initially {@code false}.
  37.      */
  38.     public FluentBitSet() {
  39.         this(new BitSet());
  40.     }

  41.     /**
  42.      * Creates a new instance for the given bit set.
  43.      *
  44.      * @param set The bit set to wrap.
  45.      */
  46.     public FluentBitSet(final BitSet set) {
  47.         this.bitSet = Objects.requireNonNull(set, "set");
  48.     }

  49.     /**
  50.      * Creates a bit set whose initial size is large enough to explicitly represent bits with indices in the range {@code 0}
  51.      * through {@code nbits-1}. All bits are initially {@code false}.
  52.      *
  53.      * @param nbits the initial size of the bit set.
  54.      * @throws NegativeArraySizeException if the specified initial size is negative.
  55.      */
  56.     public FluentBitSet(final int nbits) {
  57.         this(new BitSet(nbits));
  58.     }

  59.     /**
  60.      * Performs a logical <b>AND</b> of this target bit set with the argument bit set. This bit set is modified so that each
  61.      * bit in it has the value {@code true} if and only if it both initially had the value {@code true} and the
  62.      * corresponding bit in the bit set argument also had the value {@code true}.
  63.      *
  64.      * @param set a bit set.
  65.      * @return {@code this} instance.
  66.      */
  67.     public FluentBitSet and(final BitSet set) {
  68.         bitSet.and(set);
  69.         return this;
  70.     }

  71.     /**
  72.      * Performs a logical <b>AND</b> of this target bit set with the argument bit set. This bit set is modified so that each
  73.      * bit in it has the value {@code true} if and only if it both initially had the value {@code true} and the
  74.      * corresponding bit in the bit set argument also had the value {@code true}.
  75.      *
  76.      * @param set a bit set.
  77.      * @return {@code this} instance.
  78.      */
  79.     public FluentBitSet and(final FluentBitSet set) {
  80.         bitSet.and(set.bitSet);
  81.         return this;
  82.     }

  83.     /**
  84.      * Clears all of the bits in this {@link BitSet} whose corresponding bit is set in the specified {@link BitSet}.
  85.      *
  86.      * @param set the {@link BitSet} with which to mask this {@link BitSet}.
  87.      * @return {@code this} instance.
  88.      */
  89.     public FluentBitSet andNot(final BitSet set) {
  90.         bitSet.andNot(set);
  91.         return this;
  92.     }

  93.     /**
  94.      * Clears all of the bits in this {@link BitSet} whose corresponding bit is set in the specified {@link BitSet}.
  95.      *
  96.      * @param set the {@link BitSet} with which to mask this {@link BitSet}.
  97.      * @return {@code this} instance.
  98.      */
  99.     public FluentBitSet andNot(final FluentBitSet set) {
  100.         this.bitSet.andNot(set.bitSet);
  101.         return this;
  102.     }

  103.     /**
  104.      * Gets the wrapped bit set.
  105.      *
  106.      * @return the wrapped bit set.
  107.      */
  108.     public BitSet bitSet() {
  109.         return bitSet;
  110.     }

  111.     /**
  112.      * Returns the number of bits set to {@code true} in this {@link BitSet}.
  113.      *
  114.      * @return the number of bits set to {@code true} in this {@link BitSet}.
  115.      */
  116.     public int cardinality() {
  117.         return bitSet.cardinality();
  118.     }

  119.     /**
  120.      * Sets all of the bits in this BitSet to {@code false}.
  121.      *
  122.      * @return {@code this} instance.
  123.      */
  124.     public FluentBitSet clear() {
  125.         bitSet.clear();
  126.         return this;
  127.     }

  128.     /**
  129.      * Sets the bits specified by the indexes to {@code false}.
  130.      *
  131.      * @param bitIndexArray the index of the bit to be cleared.
  132.      * @throws IndexOutOfBoundsException if the specified index is negative.
  133.      * @return {@code this} instance.
  134.      */
  135.     public FluentBitSet clear(final int... bitIndexArray) {
  136.         for (final int e : bitIndexArray) {
  137.             this.bitSet.clear(e);
  138.         }
  139.         return this;
  140.     }

  141.     /**
  142.      * Sets the bit specified by the index to {@code false}.
  143.      *
  144.      * @param bitIndex the index of the bit to be cleared.
  145.      * @throws IndexOutOfBoundsException if the specified index is negative.
  146.      * @return {@code this} instance.
  147.      */
  148.     public FluentBitSet clear(final int bitIndex) {
  149.         bitSet.clear(bitIndex);
  150.         return this;
  151.     }

  152.     /**
  153.      * Sets the bits from the specified {@code fromIndex} (inclusive) to the specified {@code toIndex} (exclusive) to
  154.      * {@code false}.
  155.      *
  156.      * @param fromIndex index of the first bit to be cleared.
  157.      * @param toIndex index after the last bit to be cleared.
  158.      * @throws IndexOutOfBoundsException if {@code fromIndex} is negative, or {@code toIndex} is negative, or
  159.      *         {@code fromIndex} is larger than {@code toIndex}.
  160.      * @return {@code this} instance.
  161.      */
  162.     public FluentBitSet clear(final int fromIndex, final int toIndex) {
  163.         bitSet.clear(fromIndex, toIndex);
  164.         return this;
  165.     }

  166.     /**
  167.      * Cloning this {@link BitSet} produces a new {@link BitSet} that is equal to it. The clone of the bit set is another
  168.      * bit set that has exactly the same bits set to {@code true} as this bit set.
  169.      *
  170.      * @return a clone of this bit set
  171.      * @see #size()
  172.      */
  173.     @Override
  174.     public Object clone() {
  175.         return new FluentBitSet((BitSet) bitSet.clone());
  176.     }

  177.     @Override
  178.     public boolean equals(final Object obj) {
  179.         if (this == obj) {
  180.             return true;
  181.         }
  182.         if (!(obj instanceof FluentBitSet)) {
  183.             return false;
  184.         }
  185.         final FluentBitSet other = (FluentBitSet) obj;
  186.         return Objects.equals(bitSet, other.bitSet);
  187.     }

  188.     /**
  189.      * Sets the bit at the specified index to the complement of its current value.
  190.      *
  191.      * @param bitIndex the index of the bit to flip.
  192.      * @throws IndexOutOfBoundsException if the specified index is negative.
  193.      * @return {@code this} instance.
  194.      */
  195.     public FluentBitSet flip(final int bitIndex) {
  196.         bitSet.flip(bitIndex);
  197.         return this;
  198.     }

  199.     /**
  200.      * Sets each bit from the specified {@code fromIndex} (inclusive) to the specified {@code toIndex} (exclusive) to the
  201.      * complement of its current value.
  202.      *
  203.      * @param fromIndex index of the first bit to flip.
  204.      * @param toIndex index after the last bit to flip.
  205.      * @throws IndexOutOfBoundsException if {@code fromIndex} is negative, or {@code toIndex} is negative, or
  206.      *         {@code fromIndex} is larger than {@code toIndex}.
  207.      * @return {@code this} instance.
  208.      */
  209.     public FluentBitSet flip(final int fromIndex, final int toIndex) {
  210.         bitSet.flip(fromIndex, toIndex);
  211.         return this;
  212.     }

  213.     /**
  214.      * Returns the value of the bit with the specified index. The value is {@code true} if the bit with the index
  215.      * {@code bitIndex} is currently set in this {@link BitSet}; otherwise, the result is {@code false}.
  216.      *
  217.      * @param bitIndex the bit index.
  218.      * @return the value of the bit with the specified index.
  219.      * @throws IndexOutOfBoundsException if the specified index is negative.
  220.      */
  221.     public boolean get(final int bitIndex) {
  222.         return bitSet.get(bitIndex);
  223.     }

  224.     /**
  225.      * Returns a new {@link BitSet} composed of bits from this {@link BitSet} from {@code fromIndex} (inclusive) to
  226.      * {@code toIndex} (exclusive).
  227.      *
  228.      * @param fromIndex index of the first bit to include.
  229.      * @param toIndex index after the last bit to include.
  230.      * @return a new {@link BitSet} from a range of this {@link BitSet}.
  231.      * @throws IndexOutOfBoundsException if {@code fromIndex} is negative, or {@code toIndex} is negative, or
  232.      *         {@code fromIndex} is larger than {@code toIndex}.
  233.      */
  234.     public FluentBitSet get(final int fromIndex, final int toIndex) {
  235.         return new FluentBitSet(bitSet.get(fromIndex, toIndex));
  236.     }

  237.     @Override
  238.     public int hashCode() {
  239.         return bitSet.hashCode();
  240.     }

  241.     /**
  242.      * Returns true if the specified {@link BitSet} has any bits set to {@code true} that are also set to {@code true} in
  243.      * this {@link BitSet}.
  244.      *
  245.      * @param set {@link BitSet} to intersect with.
  246.      * @return boolean indicating whether this {@link BitSet} intersects the specified {@link BitSet}.
  247.      */
  248.     public boolean intersects(final BitSet set) {
  249.         return bitSet.intersects(set);
  250.     }

  251.     /**
  252.      * Returns true if the specified {@link BitSet} has any bits set to {@code true} that are also set to {@code true} in
  253.      * this {@link BitSet}.
  254.      *
  255.      * @param set {@link BitSet} to intersect with.
  256.      * @return boolean indicating whether this {@link BitSet} intersects the specified {@link BitSet}.
  257.      */
  258.     public boolean intersects(final FluentBitSet set) {
  259.         return bitSet.intersects(set.bitSet);
  260.     }

  261.     /**
  262.      * Returns true if this {@link BitSet} contains no bits that are set to {@code true}.
  263.      *
  264.      * @return boolean indicating whether this {@link BitSet} is empty.
  265.      */
  266.     public boolean isEmpty() {
  267.         return bitSet.isEmpty();
  268.     }

  269.     /**
  270.      * Returns the "logical size" of this {@link BitSet}: the index of the highest set bit in the {@link BitSet} plus one.
  271.      * Returns zero if the {@link BitSet} contains no set bits.
  272.      *
  273.      * @return the logical size of this {@link BitSet}.
  274.      */
  275.     public int length() {
  276.         return bitSet.length();
  277.     }

  278.     /**
  279.      * Returns the index of the first bit that is set to {@code false} that occurs on or after the specified starting index.
  280.      *
  281.      * @param fromIndex the index to start checking from (inclusive).
  282.      * @return the index of the next clear bit.
  283.      * @throws IndexOutOfBoundsException if the specified index is negative.
  284.      */
  285.     public int nextClearBit(final int fromIndex) {
  286.         return bitSet.nextClearBit(fromIndex);
  287.     }

  288.     /**
  289.      * Returns the index of the first bit that is set to {@code true} that occurs on or after the specified starting index.
  290.      * If no such bit exists then {@code -1} is returned.
  291.      * <p>
  292.      * To iterate over the {@code true} bits in a {@link BitSet}, use the following loop:
  293.      * </p>
  294.      *
  295.      * <pre>
  296.      * {@code
  297.      * for (int i = bs.nextSetBit(0); i >= 0; i = bs.nextSetBit(i+1)) {
  298.      *     // operate on index i here
  299.      *     if (i == Integer.MAX_VALUE) {
  300.      *         break; // or (i+1) would overflow
  301.      *     }
  302.      * }}
  303.      * </pre>
  304.      *
  305.      * @param fromIndex the index to start checking from (inclusive).
  306.      * @return the index of the next set bit, or {@code -1} if there is no such bit.
  307.      * @throws IndexOutOfBoundsException if the specified index is negative.
  308.      */
  309.     public int nextSetBit(final int fromIndex) {
  310.         return bitSet.nextSetBit(fromIndex);
  311.     }

  312.     /**
  313.      * Performs a logical <b>OR</b> of this bit set with the bit set argument. This bit set is modified so that a bit in it
  314.      * has the value {@code true} if and only if it either already had the value {@code true} or the corresponding bit in
  315.      * the bit set argument has the value {@code true}.
  316.      *
  317.      * @param set a bit set.
  318.      * @return {@code this} instance.
  319.      */
  320.     public FluentBitSet or(final BitSet set) {
  321.         bitSet.or(set);
  322.         return this;
  323.     }

  324.     /**
  325.      * Performs a logical <b>OR</b> of this bit set with the bit set arguments. This bit set is modified so that a bit in it
  326.      * has the value {@code true} if and only if it either already had the value {@code true} or the corresponding bit in
  327.      * the bit set argument has the value {@code true}.
  328.      *
  329.      * @param set a bit set.
  330.      * @return {@code this} instance.
  331.      */
  332.     public FluentBitSet or(final FluentBitSet... set) {
  333.         for (final FluentBitSet e : set) {
  334.             this.bitSet.or(e.bitSet);
  335.         }
  336.         return this;
  337.     }

  338.     /**
  339.      * Performs a logical <b>OR</b> of this bit set with the bit set argument. This bit set is modified so that a bit in it
  340.      * has the value {@code true} if and only if it either already had the value {@code true} or the corresponding bit in
  341.      * the bit set argument has the value {@code true}.
  342.      *
  343.      * @param set a bit set.
  344.      * @return {@code this} instance.
  345.      */
  346.     public FluentBitSet or(final FluentBitSet set) {
  347.         this.bitSet.or(set.bitSet);
  348.         return this;
  349.     }

  350.     /**
  351.      * Returns the index of the nearest bit that is set to {@code false} that occurs on or before the specified starting
  352.      * index. If no such bit exists, or if {@code -1} is given as the starting index, then {@code -1} is returned.
  353.      *
  354.      * @param fromIndex the index to start checking from (inclusive).
  355.      * @return the index of the previous clear bit, or {@code -1} if there is no such bit.
  356.      * @throws IndexOutOfBoundsException if the specified index is less than {@code -1}.
  357.      */
  358.     public int previousClearBit(final int fromIndex) {
  359.         return bitSet.previousClearBit(fromIndex);
  360.     }

  361.     /**
  362.      * Returns the index of the nearest bit that is set to {@code true} that occurs on or before the specified starting
  363.      * index. If no such bit exists, or if {@code -1} is given as the starting index, then {@code -1} is returned.
  364.      *
  365.      * <p>
  366.      * To iterate over the {@code true} bits in a {@link BitSet}, use the following loop:
  367.      *
  368.      * <pre>
  369.      *  {@code
  370.      * for (int i = bs.length(); (i = bs.previousSetBit(i-1)) >= 0; ) {
  371.      *     // operate on index i here
  372.      * }}
  373.      * </pre>
  374.      *
  375.      * @param fromIndex the index to start checking from (inclusive)
  376.      * @return the index of the previous set bit, or {@code -1} if there is no such bit
  377.      * @throws IndexOutOfBoundsException if the specified index is less than {@code -1}
  378.      */
  379.     public int previousSetBit(final int fromIndex) {
  380.         return bitSet.previousSetBit(fromIndex);
  381.     }

  382.     /**
  383.      * Sets the bit at the specified indexes to {@code true}.
  384.      *
  385.      * @param bitIndexArray a bit index array.
  386.      * @throws IndexOutOfBoundsException if the specified index is negative.
  387.      * @return {@code this} instance.
  388.      */
  389.     public FluentBitSet set(final int... bitIndexArray) {
  390.         for (final int e : bitIndexArray) {
  391.             bitSet.set(e);
  392.         }
  393.         return this;
  394.     }

  395.     /**
  396.      * Sets the bit at the specified index to {@code true}.
  397.      *
  398.      * @param bitIndex a bit index
  399.      * @throws IndexOutOfBoundsException if the specified index is negative
  400.      * @return {@code this} instance.
  401.      */
  402.     public FluentBitSet set(final int bitIndex) {
  403.         bitSet.set(bitIndex);
  404.         return this;
  405.     }

  406.     /**
  407.      * Sets the bit at the specified index to the specified value.
  408.      *
  409.      * @param bitIndex a bit index.
  410.      * @param value a boolean value to set.
  411.      * @throws IndexOutOfBoundsException if the specified index is negative.
  412.      * @return {@code this} instance.
  413.      */
  414.     public FluentBitSet set(final int bitIndex, final boolean value) {
  415.         bitSet.set(bitIndex, value);
  416.         return this;
  417.     }

  418.     /**
  419.      * Sets the bits from the specified {@code fromIndex} (inclusive) to the specified {@code toIndex} (exclusive) to
  420.      * {@code true}.
  421.      *
  422.      * @param fromIndex index of the first bit to be set.
  423.      * @param toIndex index after the last bit to be set.
  424.      * @throws IndexOutOfBoundsException if {@code fromIndex} is negative, or {@code toIndex} is negative, or
  425.      *         {@code fromIndex} is larger than {@code toIndex}.
  426.      * @return {@code this} instance.
  427.      */
  428.     public FluentBitSet set(final int fromIndex, final int toIndex) {
  429.         bitSet.set(fromIndex, toIndex);
  430.         return this;
  431.     }

  432.     /**
  433.      * Sets the bits from the specified {@code fromIndex} (inclusive) to the specified {@code toIndex} (exclusive) to the
  434.      * specified value.
  435.      *
  436.      * @param fromIndex index of the first bit to be set.
  437.      * @param toIndex index after the last bit to be set.
  438.      * @param value value to set the selected bits to.
  439.      * @throws IndexOutOfBoundsException if {@code fromIndex} is negative, or {@code toIndex} is negative, or
  440.      *         {@code fromIndex} is larger than {@code toIndex}.
  441.      * @return {@code this} instance.
  442.      */
  443.     public FluentBitSet set(final int fromIndex, final int toIndex, final boolean value) {
  444.         bitSet.set(fromIndex, toIndex, value);
  445.         return this;
  446.     }

  447.     /**
  448.      * Sets the bits from the specified {@code fromIndex} (inclusive) to the specified {@code toIndex} (inclusive) to
  449.      * {@code true}.
  450.      *
  451.      * @param fromIndex index of the first bit to be set
  452.      * @param toIndex index of the last bit to be set
  453.      * @throws IndexOutOfBoundsException if {@code fromIndex} is negative, or {@code toIndex} is negative, or
  454.      *         {@code fromIndex} is larger than {@code toIndex}
  455.      * @return {@code this} instance.
  456.      */
  457.     public FluentBitSet setInclusive(final int fromIndex, final int toIndex) {
  458.         bitSet.set(fromIndex, toIndex + 1);
  459.         return this;
  460.     }

  461.     /**
  462.      * Returns the number of bits of space actually in use by this {@link BitSet} to represent bit values. The maximum
  463.      * element in the set is the size - 1st element.
  464.      *
  465.      * @return the number of bits currently in this bit set.
  466.      */
  467.     public int size() {
  468.         return bitSet.size();
  469.     }

  470.     /**
  471.      * Returns a stream of indices for which this {@link BitSet} contains a bit in the set state. The indices are returned
  472.      * in order, from lowest to highest. The size of the stream is the number of bits in the set state, equal to the value
  473.      * returned by the {@link #cardinality()} method.
  474.      *
  475.      * <p>
  476.      * The bit set must remain constant during the execution of the terminal stream operation. Otherwise, the result of the
  477.      * terminal stream operation is undefined.
  478.      * </p>
  479.      *
  480.      * @return a stream of integers representing set indices.
  481.      * @since 1.8
  482.      */
  483.     public IntStream stream() {
  484.         return bitSet.stream();
  485.     }

  486.     /**
  487.      * Returns a new byte array containing all the bits in this bit set.
  488.      *
  489.      * <p>
  490.      * More precisely, if:
  491.      * </p>
  492.      * <ol>
  493.      * <li>{@code byte[] bytes = s.toByteArray();}</li>
  494.      * <li>then {@code bytes.length == (s.length()+7)/8} and</li>
  495.      * <li>{@code s.get(n) == ((bytes[n/8] & (1<<(n%8))) != 0)}</li>
  496.      * <li>for all {@code n < 8 * bytes.length}.</li>
  497.      * </ol>
  498.      *
  499.      * @return a byte array containing a little-endian representation of all the bits in this bit set
  500.      */
  501.     public byte[] toByteArray() {
  502.         return bitSet.toByteArray();
  503.     }

  504.     /**
  505.      * Returns a new byte array containing all the bits in this bit set.
  506.      *
  507.      * <p>
  508.      * More precisely, if:
  509.      * </p>
  510.      * <ol>
  511.      * <li>{@code long[] longs = s.toLongArray();}</li>
  512.      * <li>then {@code longs.length == (s.length()+63)/64} and</li>
  513.      * <li>{@code s.get(n) == ((longs[n/64] & (1L<<(n%64))) != 0)}</li>
  514.      * <li>for all {@code n < 64 * longs.length}.</li>
  515.      * </ol>
  516.      *
  517.      * @return a byte array containing a little-endian representation of all the bits in this bit set
  518.      */
  519.     public long[] toLongArray() {
  520.         return bitSet.toLongArray();
  521.     }

  522.     @Override
  523.     public String toString() {
  524.         return bitSet.toString();
  525.     }

  526.     /**
  527.      * Performs a logical <b>XOR</b> of this bit set with the bit set argument. This bit set is modified so that a bit in it
  528.      * has the value {@code true} if and only if one of the following statements holds:
  529.      * <ul>
  530.      * <li>The bit initially has the value {@code true}, and the corresponding bit in the argument has the value
  531.      * {@code false}.
  532.      * <li>The bit initially has the value {@code false}, and the corresponding bit in the argument has the value
  533.      * {@code true}.
  534.      * </ul>
  535.      *
  536.      * @param set a bit set
  537.      * @return {@code this} instance.
  538.      */
  539.     public FluentBitSet xor(final BitSet set) {
  540.         bitSet.xor(set);
  541.         return this;
  542.     }

  543.     /**
  544.      * Performs a logical <b>XOR</b> of this bit set with the bit set argument. This bit set is modified so that a bit in it
  545.      * has the value {@code true} if and only if one of the following statements holds:
  546.      * <ul>
  547.      * <li>The bit initially has the value {@code true}, and the corresponding bit in the argument has the value
  548.      * {@code false}.
  549.      * <li>The bit initially has the value {@code false}, and the corresponding bit in the argument has the value
  550.      * {@code true}.
  551.      * </ul>
  552.      *
  553.      * @param set a bit set
  554.      * @return {@code this} instance.
  555.      */
  556.     public FluentBitSet xor(final FluentBitSet set) {
  557.         bitSet.xor(set.bitSet);
  558.         return this;
  559.     }

  560. }