YTSFlix_Go/vendor/github.com/anacrolix/missinggo/bitmap/bitmap.go
2018-11-04 15:58:15 +01:00

172 lines
3.0 KiB
Go

// Package bitmap provides a []bool/bitmap implementation with standardized
// iteration. Bitmaps are the equivalent of []bool, with improved compression
// for runs of similar values, and faster operations on ranges and the like.
package bitmap
import (
"math"
"github.com/RoaringBitmap/roaring"
"github.com/anacrolix/missinggo/iter"
)
const MaxInt = -1
type BitIndex = int
type Interface interface {
Len() int
}
// Bitmaps store the existence of values in [0,math.MaxUint32] more
// efficiently than []bool. The empty value starts with no bits set.
type Bitmap struct {
rb *roaring.Bitmap
}
var ToEnd int = -1
// The number of set bits in the bitmap. Also known as cardinality.
func (me *Bitmap) Len() int {
if me.rb == nil {
return 0
}
return int(me.rb.GetCardinality())
}
func (me Bitmap) ToSortedSlice() (ret []int) {
if me.rb == nil {
return
}
for _, ui32 := range me.rb.ToArray() {
ret = append(ret, int(int32(ui32)))
}
return
}
func (me *Bitmap) lazyRB() *roaring.Bitmap {
if me.rb == nil {
me.rb = roaring.NewBitmap()
}
return me.rb
}
func (me Bitmap) Iter(cb iter.Callback) {
me.IterTyped(func(i int) bool {
return cb(i)
})
}
// Returns true if all values were traversed without early termination.
func (me Bitmap) IterTyped(f func(int) bool) bool {
if me.rb == nil {
return true
}
it := me.rb.Iterator()
for it.HasNext() {
if !f(int(it.Next())) {
return false
}
}
return true
}
func checkInt(i BitIndex) {
if i < math.MinInt32 || i > math.MaxInt32 {
panic("out of bounds")
}
}
func (me *Bitmap) Add(is ...BitIndex) {
rb := me.lazyRB()
for _, i := range is {
checkInt(i)
rb.AddInt(i)
}
}
func (me *Bitmap) AddRange(begin, end BitIndex) {
if begin >= end {
return
}
me.lazyRB().AddRange(uint64(begin), uint64(end))
}
func (me *Bitmap) Remove(i BitIndex) bool {
if me.rb == nil {
return false
}
return me.rb.CheckedRemove(uint32(i))
}
func (me *Bitmap) Union(other Bitmap) {
me.lazyRB().Or(other.lazyRB())
}
func (me *Bitmap) Contains(i int) bool {
if me.rb == nil {
return false
}
return me.rb.Contains(uint32(i))
}
func (me *Bitmap) Sub(other Bitmap) {
if other.rb == nil {
return
}
if me.rb == nil {
return
}
me.rb.AndNot(other.rb)
}
func (me *Bitmap) Clear() {
if me.rb == nil {
return
}
me.rb.Clear()
}
func (me Bitmap) Copy() (ret Bitmap) {
ret = me
if ret.rb != nil {
ret.rb = ret.rb.Clone()
}
return
}
func (me *Bitmap) FlipRange(begin, end BitIndex) {
me.lazyRB().FlipInt(begin, end)
}
func (me *Bitmap) Get(bit BitIndex) bool {
return me.rb != nil && me.rb.ContainsInt(bit)
}
func (me *Bitmap) Set(bit BitIndex, value bool) {
if value {
me.lazyRB().AddInt(bit)
} else {
if me.rb != nil {
me.rb.Remove(uint32(bit))
}
}
}
func (me *Bitmap) RemoveRange(begin, end BitIndex) *Bitmap {
if me.rb == nil {
return me
}
rangeEnd := uint64(end)
if end == ToEnd {
rangeEnd = 0x100000000
}
me.rb.RemoveRange(uint64(begin), rangeEnd)
return me
}
func (me Bitmap) IsEmpty() bool {
return me.rb == nil || me.rb.IsEmpty()
}