Skip to content

Commit

Permalink
Merge pull request #35 from emirpasic/heap_bulk_push
Browse files Browse the repository at this point in the history
Heap Bulk Push Optimization
  • Loading branch information
emirpasic authored Sep 7, 2016
2 parents 549b638 + 4ea8571 commit 8e82839
Show file tree
Hide file tree
Showing 3 changed files with 36 additions and 9 deletions.
4 changes: 1 addition & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -710,9 +710,7 @@ func main() {
return -utils.IntComparator(a, b)
}
heap = binaryheap.NewWith(inverseIntComparator) // empty (min-heap)
heap.Push(2) // 2
heap.Push(3) // 3, 2
heap.Push(1) // 3, 2, 1
heap.Push(2, 3, 1) // 3, 2, 1 (bulk optimized)
heap.Values() // 3, 2, 1
}
```
Expand Down
28 changes: 22 additions & 6 deletions trees/binaryheap/binaryheap.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,9 +45,20 @@ func NewWithStringComparator() *Heap {
}

// Push adds a value onto the heap and bubbles it up accordingly.
func (heap *Heap) Push(value interface{}) {
heap.list.Add(value)
heap.bubbleUp()
func (heap *Heap) Push(values ...interface{}) {
if len(values) == 1 {
heap.list.Add(values[0])
heap.bubbleUp()
} else {
// Reference: https://en.wikipedia.org/wiki/Binary_heap#Building_a_heap
for _, value := range values {
heap.list.Add(value)
}
size := heap.list.Size()/2 + 1
for i := size; i >= 0; i-- {
heap.bubbleDownIndex(i)
}
}
}

// Pop removes top element on heap and returns it, or nil if heap is empty.
Expand Down Expand Up @@ -101,10 +112,15 @@ func (heap *Heap) String() string {
return str
}

// Performs the "bubble down" operation. This is to place the element that is at the
// root of the heap in its correct place so that the heap maintains the min/max-heap order property.
// Performs the "bubble down" operation. This is to place the element that is at the root
// of the heap in its correct place so that the heap maintains the min/max-heap order property.
func (heap *Heap) bubbleDown() {
index := 0
heap.bubbleDownIndex(0)
}

// Performs the "bubble down" operation. This is to place the element that is at the index
// of the heap in its correct place so that the heap maintains the min/max-heap order property.
func (heap *Heap) bubbleDownIndex(index int) {
size := heap.list.Size()
for leftIndex := index<<1 + 1; leftIndex < size; leftIndex = index<<1 + 1 {
rightIndex := index<<1 + 2
Expand Down
13 changes: 13 additions & 0 deletions trees/binaryheap/binaryheap_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,19 @@ func TestBinaryHeapPush(t *testing.T) {
}
}

func TestBinaryHeapPushBulk(t *testing.T) {
heap := NewWithIntComparator()

heap.Push(15, 20, 3, 1, 2)

if actualValue := heap.Values(); actualValue[0].(int) != 1 || actualValue[1].(int) != 2 || actualValue[2].(int) != 3 {
t.Errorf("Got %v expected %v", actualValue, "[1,2,3]")
}
if actualValue, ok := heap.Pop(); actualValue != 1 || !ok {
t.Errorf("Got %v expected %v", actualValue, 1)
}
}

func TestBinaryHeapPop(t *testing.T) {
heap := NewWithIntComparator()

Expand Down

0 comments on commit 8e82839

Please sign in to comment.