priority_queue.odin 2.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. package container
  2. Priority_Queue :: struct($T: typeid) {
  3. data: Array(T),
  4. len: int,
  5. priority: proc(item: T) -> int,
  6. }
  7. priority_queue_init_none :: proc(q: ^$Q/Priority_Queue($T), f: proc(item: T) -> int, allocator := context.allocator) {
  8. queue_init_len(q, f, 0, allocator)
  9. }
  10. priority_queue_init_len :: proc(q: ^$Q/Priority_Queue($T), f: proc(item: T) -> int, len: int, allocator := context.allocator) {
  11. queue_init_len_cap(q, f, 0, 16, allocator)
  12. }
  13. priority_queue_init_len_cap :: proc(q: ^$Q/Priority_Queue($T), f: proc(item: T) -> int, len: int, cap: int, allocator := context.allocator) {
  14. array_init(&q.data, len, cap, allocator)
  15. q.len = len
  16. q.priority = f
  17. }
  18. priority_queue_init :: proc{priority_queue_init_none, priority_queue_init_len, priority_queue_init_len_cap}
  19. priority_queue_delete :: proc(q: $Q/Priority_Queue($T)) {
  20. array_delete(q.data)
  21. }
  22. priority_queue_clear :: proc(q: ^$Q/Priority_Queue($T)) {
  23. q.len = 0
  24. }
  25. priority_queue_len :: proc(q: $Q/Priority_Queue($T)) -> int {
  26. return q.len
  27. }
  28. priority_queue_cap :: proc(q: $Q/Priority_Queue($T)) -> int {
  29. return array_cap(q.data)
  30. }
  31. priority_queue_space :: proc(q: $Q/Priority_Queue($T)) -> int {
  32. return array_len(q.data) - q.len
  33. }
  34. priority_queue_reserve :: proc(q: ^$Q/Priority_Queue($T), capacity: int) {
  35. if capacity > q.len {
  36. array_resize(&q.data, new_capacity)
  37. }
  38. }
  39. priority_queue_resize :: proc(q: ^$Q/Priority_Queue($T), length: int) {
  40. if length > q.len {
  41. array_resize(&q.data, new_capacity)
  42. }
  43. q.len = length
  44. }
  45. _priority_queue_grow :: proc(q: ^$Q/Priority_Queue($T), min_capacity: int = 0) {
  46. new_capacity := max(array_len(q.data)*2 + 8, min_capacity)
  47. array_resize(&q.data, new_capacity)
  48. }
  49. priority_queue_push :: proc(q: ^$Q/Priority_Queue($T), item: T) {
  50. if array_len(q.data) - q.len == 0 {
  51. _priority_queue_grow(q)
  52. }
  53. s := array_slice(q.data)
  54. s[q.len] = item
  55. i := q.len
  56. for i > 0 {
  57. p := (i - 1) / 2
  58. if q.priority(s[p]) <= q.priority(item) {
  59. break
  60. }
  61. s[i] = s[p]
  62. i = p
  63. }
  64. q.len += 1
  65. if q.len > 0 {
  66. s[i] = item
  67. }
  68. }
  69. priority_queue_pop :: proc(q: ^$Q/Priority_Queue($T)) -> T {
  70. assert(q.len > 0)
  71. s := array_slice(q.data)
  72. min := s[0]
  73. root := s[q.len-1]
  74. q.len -= 1
  75. i := 0
  76. for i * 2 + 1 < q.len {
  77. a := i * 2 + 1
  78. b := i * 2 + 2
  79. c := b < q.len && q.priority(s[b]) < q.priority(s[a]) ? b : a
  80. if q.priority(s[c]) >= q.priority(root) {
  81. break
  82. }
  83. s[i] = s[c]
  84. i = c
  85. }
  86. if q.len > 0 {
  87. s[i] = root
  88. }
  89. return min
  90. }
  91. priority_queue_peek :: proc(q: ^$Q/Priority_Queue($T)) -> T {
  92. assert(q.len > 0)
  93. s := array_slice(q.data)
  94. return s[0]
  95. }