Browse Source

Change the implementation of `Priority_Queue` to have a better interface that allows for a `less` and `swap` procedure

gingerBill 3 years ago
parent
commit
c7ff296bef
1 changed files with 102 additions and 104 deletions
  1. 102 104
      core/container/priority_queue/priority_queue.odin

+ 102 - 104
core/container/priority_queue/priority_queue.odin

@@ -3,138 +3,136 @@ package container_priority_queue
 import "core:builtin"
 import "core:builtin"
 
 
 Priority_Queue :: struct($T: typeid) {
 Priority_Queue :: struct($T: typeid) {
-	data:     [dynamic]T,
-	len:      int,
-	priority: proc(item: T) -> int,
+	queue: [dynamic]T,
+	
+	less:  proc(a, b: T) -> bool,
+	swap:  proc(q: []T, i, j: int),
 }
 }
 
 
 DEFAULT_CAPACITY :: 16
 DEFAULT_CAPACITY :: 16
 
 
-init_none :: proc(q: ^$Q/Priority_Queue($T), f: proc(item: T) -> int, allocator := context.allocator) {
-	init_len(q, f, 0, allocator)
-}
-init_len :: proc(q: ^$Q/Priority_Queue($T), f: proc(item: T) -> int, len: int, allocator := context.allocator) {
-	init_len_cap(q, f, 0, DEFAULT_CAPACITY, allocator)
-}
-init_len_cap :: proc(q: ^$Q/Priority_Queue($T), f: proc(item: T) -> int, len: int, cap: int, allocator := context.allocator) {
-	if q.data.allocator.procedure == nil {
-		q.data.allocator = allocator
+init :: proc(pq: ^$Q/Priority_Queue($T), less: proc(a, b: T) -> bool, swap: proc(q: []T, i, j: int), capacity := DEFAULT_CAPACITY, allocator := context.allocator) {
+	if pq.queue.allocator.procedure == nil {
+		pq.queue.allocator = allocator
+	}
+	reserve(pq, capacity)
+	pq.less = less
+	pq.swap = swap
+}
+
+init_from_dynamic_array :: proc(pq: ^$Q/Priority_Queue($T), queue: [dynamic]T, less: proc(a, b: T) -> bool, swap: proc(q: []T, i, j: int)) {
+	pq.queue = queue
+	pq.less = less
+	pq.swap = swap
+	n := builtin.len(pq.queue)
+	for i := n/2 - 1; i >= 0; i -= 1 {
+		_shift_down(pq, i, n)
 	}
 	}
-	builtin.resize(&q.data, cap)
-	q.len = len
-	q.priority = f
 }
 }
 
 
-init :: proc{init_none, init_len, init_len_cap}
-
-
-delete :: proc(q: $Q/Priority_Queue($T)) {
-	builtin.delete(q.data)
+destroy :: proc(pq: ^$Q/Priority_Queue($T)) {
+	clear(pq)
+	delete(pq.queue)
 }
 }
 
 
-clear :: proc(q: ^$Q/Priority_Queue($T)) {
-	q.len = 0
+reserve :: proc(pq: ^$Q/Priority_Queue($T), capacity: int) {
+	builtin.reserve(&pq.queue, capacity)
 }
 }
-
-len :: proc(q: $Q/Priority_Queue($T)) -> int {
-	return q.len
+clear :: proc(pq: ^$Q/Priority_Queue($T)) {
+	builtin.clear(&pq.queue)
 }
 }
-
-cap :: proc(q: $Q/Priority_Queue($T)) -> int {
-	return builtin.cap(q.data)
+len :: proc(pq: $Q/Priority_Queue($T)) -> int {
+	return builtin.len(pq.queue)
 }
 }
-
-space :: proc(q: $Q/Priority_Queue($T)) -> int {
-	return builtin.len(q.data) - q.len
+cap :: proc(pq: $Q/Priority_Queue($T)) -> int {
+	return builtin.cap(pq.queue)
 }
 }
 
 
-reserve :: proc(q: ^$Q/Priority_Queue($T), capacity: int) {
-	if capacity > q.len {
-		builtin.resize(&q.data, capacity)
+_shift_down :: proc(pq: ^$Q/Priority_Queue($T), i0, n: int) -> bool {
+	// O(n log n)
+	i := i0
+	j, j1, j2: int
+	if 0 > i || i > n {
+		return false
 	}
 	}
-}
-
-resize :: proc(q: ^$Q/Priority_Queue($T), length: int) {
-	if length > q.len {
-		builtin.resize(&q.data, length)
+	
+	queue := pq.queue[:]
+	
+	for {
+		j1 := 2*i + 1
+		if 0 > j1 || j1 >= n {
+			break
+		}
+		j, j2 = j1, j1+1
+		if j1 < n && pq.less(queue[j2], queue[j1])  {
+			j1 = j2
+		}
+		if !pq.less(queue[i], queue[j]) {
+			break
+		}
+		
+		pq.swap(queue, i, j)
+		i = j
 	}
 	}
-	q.len = length
-}
-
-_grow :: proc(q: ^$Q/Priority_Queue($T), min_capacity: int = 8) {
-	new_capacity := max(builtin.len(q.data)*2, min_capacity, 1)
-	builtin.resize(&q.data, new_capacity)
+	return i > i0
 }
 }
 
 
-
-push :: proc(q: ^$Q/Priority_Queue($T), item: T) {
-	if builtin.len(q.data) - q.len == 0 {
-		_grow(q)
-	}
-
-	s := q.data[:]
-	s[q.len] = item
-
-	i := q.len
-	for i > 0 {
-		p := (i - 1) / 2
-		if q.priority(s[p]) <= q.priority(item) { 
-			break 
+_shift_up :: proc(pq: ^$Q/Priority_Queue($T), j: int) {
+	j := j
+	queue := pq.queue[:]
+	n := builtin.len(queue)
+	for 0 <= j && j < n {
+		i := (j-1)/2
+		if i == j || !pq.less(queue[j], queue[i]) {
+			break
 		}
 		}
-		s[i] = s[p]
-		i = p
+		pq.swap(queue, i, j)
+		j = i
 	}
 	}
-
-	q.len += 1
-	if q.len > 0 { 
-		s[i] = item 
-	} 
 }
 }
 
 
-pop :: proc(q: ^$Q/Priority_Queue($T), loc := #caller_location) -> T {
-	val, ok := pop_safe(q)
-	assert(condition=ok, loc=loc)
-	return val
+// NOTE(bill): When an element at index 'i' has changed its value, this will fix the
+// the heap ordering. This is using a basic "heapsort" with shift up and a shift down parts.
+fix :: proc(pq: ^$Q/Priority_Queue($T), i: int) {
+	if !_shift_down(pq, i, builtin.len(pq.queue)) {
+		_shift_up(pq, i)
+	}
 }
 }
 
 
+push :: proc(pq: ^$Q/Priority_Queue($T), value: T) {
+	append(&pq.queue, value)
+	_shift_up(pq, builtin.len(pq.queue)-1)
+}
 
 
-pop_safe :: proc(q: ^$Q/Priority_Queue($T)) -> (T, bool) {
-	if q.len > 0 {
-		s := q.data[:]
-		min := s[0]
-		root := s[q.len-1]
-		q.len -= 1
-
-		i := 0
-		for i * 2 + 1 < q.len {
-			a := i * 2 + 1
-			b := i * 2 + 2
-			c := b < q.len && q.priority(s[b]) < q.priority(s[a]) ? b : a
-
-			if q.priority(s[c]) >= q.priority(root) {
-				break
-			}
-			s[i] = s[c]
-			i = c
-		}
+pop :: proc(pq: ^$Q/Priority_Queue($T), loc := #caller_location) -> (value: T) {
+	assert(condition=builtin.len(pq.queue)>0, loc=loc)
+	
+	n := builtin.len(pq.queue)-1
+	pq.swap(pq.queue[:], 0, n)
+	_shift_down(pq, 0, n)
+	return builtin.pop(&pq.queue)
+}
 
 
-		if q.len > 0 {
-			s[i] = root
-		}
-		return min, true
+pop_safe :: proc(pq: ^$Q/Priority_Queue($T), loc := #caller_location) -> (value: T, ok: bool) {
+	if builtin.len(pq.queue) > 0 {
+		n := builtin.len(pq.queue)-1
+		pq.swap(pq.queue[:], 0, n)
+		_shift_down(pq, 0, n)
+		return builtin.pop_safe(&pq.queue)
 	}
 	}
-	return T{}, false
+	return
 }
 }
 
 
-peek :: proc(q: ^$Q/Priority_Queue($T), loc := #caller_location) -> T {
-	assert(condition=q.len > 0, loc=loc)
-
-	return q.data[0]
+remove :: proc(pq: ^$Q/Priority_Queue($T), i: int) -> (value: T, ok: bool) {
+	n := builtin.len(pq.queue)
+	if 0 <= i && i < n {
+		if n != i {
+			pq.swap(pq.queue[:], i, n)
+			_shift_down(pq, i, n)
+			_shift_up(pq, i)
+		}
+		value, ok = builtin.pop_safe(&pq.queue)
+	}
+	return
 }
 }
 
 
-peek_safe :: proc(q: ^$Q/Priority_Queue($T)) -> (T, bool) {
-	if q.len > 0 {
-		return q.data[0], true
-	}
-	return T{}, false
-}