java.util.BitSet — set() doesn't work as expected

后端 未结 6 1360
借酒劲吻你
借酒劲吻你 2021-01-18 01:17

Am I missing something painfully obvious? Or does just nobody in the world actually use java.util.BitSet?

The following test fails:

@Test
public voi         


        
相关标签:
6条回答
  • 2021-01-18 01:33

    Given that the bitset is backed by a long[], the minimum size is 64 (because 1 long is 64 bits). The size gets incremented by a multiple of 64 and for some reason, they have not maintained the # of bits you intended to represent when you use the constructor that takes an int.

    0 讨论(0)
  • 2021-01-18 01:37

    // Abhay Dandekar

    import java.util.BitSet;
    
    public class TestBitSet {
    
        public static void main(String[] args) {
    
            BitSet bitSet = new BitSet();
            System.out.println("State 0 : " + bitSet.size() + " : " + bitSet.length() );
    
            bitSet.set(0, true);
            bitSet.set(1, true);
            System.out.println("State 1 : " + bitSet.size() + " : " + bitSet.length() );
    
            bitSet.set(2, false);
            bitSet.set(3, false);
            System.out.println("State 2 : " + bitSet.size() + " : " + bitSet.length() );
    
            bitSet.set(4, true);
            System.out.println("State 3 : " + bitSet.size() + " : " + bitSet.length() );
    
        }
    }
    

    A simple java program to show what happens inside. Some points to note :

    1. BitSet is backed by a long

    2. All the default values are false

    3. While returning the length, it returns the index+1 of the highest "true" value in the set.

    The output below should be able to explain itself :

    State 0 : 64 : 0
    
    State 1 : 64 : 2
    
    State 2 : 64 : 2
    
    State 3 : 64 : 5
    

    So points to conclude :

    1. Do not use the length to conclude the no of bits modified

    2. Can be used in scenarios like bloom filters. More on bloom filters can be googled .. ;)

    Hope this helps

    Regards,

    Abhay Dandekar

    0 讨论(0)
  • 2021-01-18 01:39

    Good Casper! Your small improvement should indeed have been present in the original BitSet java def! I also suggest this (append() and concat() are useful for various usages)

    import java.util.BitSet;
    
    public class fixBitSet extends BitSet {
    
      public int fsize = 0;
    
      public void set(int k, boolean value) {
        if (k >= fsize)
          fsize = k + 1;
        super.set(k, value);
      }
    
      public void append(fixBitSet bs) {
        for (int k = 0; k < bs.fsize; k++)
          super.set(fsize + k, bs.get(k));
        fsize += bs.fsize;
      }
    
      public static fixBitSet concat(fixBitSet[] vbs) {
        final fixBitSet bs = new fixBitSet();
        for (fixBitSet xbs : vbs)
          bs.append(xbs);
        return (bs);
      }
    
    }
    
    0 讨论(0)
  • 2021-01-18 01:55

    People do use BitSet; however, they use it for something other than what you intend. It's probably best to think of BitSet as a very compact, memory-efficient form of Set<Integer> that has the peculiar property that you can't put negative numbers into it.

    It's very common with BitSets to use them in the pattern of

    for (int id = set.nextSetBit(0); id >= 0; id = set.nextSetBit(id + 1)) {
      // do stuff to a set index
    }
    

    after you do something to fill them up. This is equivalent to iterating over the elements of the Set.

    0 讨论(0)
  • 2021-01-18 01:59

    You highest bit set (as in "set to 1") is Bit 0. So the length should be 1.

    See the JavaDoc for length:

    public int length()

    Returns the "logical size" of this BitSet: the index of the highest set bit in the BitSet plus one. Returns zero if the BitSet contains no set bits.

    Maybe you're looking for size although it's possible that might be higher than two if bits are allocated at a certain resolution (say 16 bit boundaries)?

    0 讨论(0)
  • 2021-01-18 02:00

    This puzzled me too, not sure of the rationale behind BitSet's current rather unexpected functionality. However since it's not final, we can use some embrace and extend tactics and do the following to get a fixed BitSet with length semantics as expected:

    import java.util.BitSet;
    
    /**
     * Variation of BitSet which does NOT interpret the highest bit synonymous with
     * its length.
     *
     * @author casper.bang@gmail.com
     */
    public class FixedBitSet extends BitSet{
    
        int fixedLength;
    
        public FixedBitSet(int fixedLength){
            super(fixedLength);
            this.fixedLength = fixedLength;
        }
    
        @Override
        public int length() {
            return fixedLength;
        }
    }
    
    0 讨论(0)
提交回复
热议问题