forked from google/or-tools
-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathinteger_pq.h
192 lines (170 loc) · 6.51 KB
/
integer_pq.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
// Copyright 2010-2024 Google LLC
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// This file contains and adjustable priority queue templated by an Element
// class that must:
// - Be efficiently copiable and storable in a std::vector<Element>.
// - Be comparable via the templated Compare class. Top() will returns
// the element with the largest priority (like std::priority_queue).
// - Implements the "int Index() const" function that must returns an integer
// that uniquely identify this particular element. Ideally this index must
// be dense in [0, max_num_elements).
//
#ifndef OR_TOOLS_UTIL_INTEGER_PQ_H_
#define OR_TOOLS_UTIL_INTEGER_PQ_H_
#include <functional>
#include <vector>
#include "ortools/base/logging.h"
namespace operations_research {
// Classic asjustable priority queue implementation. It behaves exactly the same
// as AdjustablePriorityQueue regarding identical elements, but it uses less
// memory and is in general slightly faster.
template <typename Element, class Compare = std::less<Element>>
class IntegerPriorityQueue {
public:
// Starts with an empty queue and reserve space for n elements.
explicit IntegerPriorityQueue(int n = 0, Compare comp = Compare())
: size_(0), less_(comp) {
Reserve(n);
}
// Increases the space reservation to n elements indices in [0, n). All
// elements passed to the other functions must have an Index() smaller than
// this n.
void Reserve(int n) {
// The heap_ starts at 1 for faster left/right child indices computation.
// This also allow us to use position 0 for element not in the queue.
heap_.resize(n + 1);
position_.resize(n, 0);
}
// Returns the number of elements currently present.
int Size() const { return size_; }
bool IsEmpty() const { return size_ == 0; }
// Removes all elements from the queue.
// TODO(user): we could make this sparse if it is needed.
void Clear() {
size_ = 0;
position_.assign(position_.size(), 0);
}
// Returns true if the element with given index is currently in the queue.
bool Contains(int index) const { return position_[index] != 0; }
// Adds the given element to the queue and set its priority.
// Preconditions: Contains(element) must be false.
void Add(Element element) {
DCHECK(!Contains(element.Index()));
SetAndIncreasePriority(++size_, element);
}
// Top() returns the top element and Pop() remove it from the queue.
// Preconditions: IsEmpty() must be false.
Element Top() const { return heap_[1]; }
void Pop() {
DCHECK(!IsEmpty());
position_[Top().Index()] = 0;
const int old_size = size_--;
if (old_size > 1) SetAndDecreasePriority(1, heap_[old_size]);
}
// Removes the element with given index from the queue.
// Preconditions: Contains(index) must be true.
void Remove(int index) {
DCHECK(Contains(index));
const int to_replace = position_[index];
position_[index] = 0;
const int old_size = size_--;
if (to_replace == old_size) return;
const Element element = heap_[old_size];
if (less_(element, heap_[to_replace])) {
SetAndDecreasePriority(to_replace, element);
} else {
SetAndIncreasePriority(to_replace, element);
}
}
// Change the priority of the given element and adjust the queue.
//
// Preconditions: Contains(element) must be true.
void ChangePriority(Element element) {
DCHECK(Contains(element.Index()));
const int i = position_[element.Index()];
if (i > 1 && less_(heap_[i >> 1], element)) {
SetAndIncreasePriority(i, element);
} else {
SetAndDecreasePriority(i, element);
}
}
// Optimized version of ChangePriority() when we know the direction.
void IncreasePriority(Element element) {
SetAndIncreasePriority(position_[element.Index()], element);
}
void DecreasePriority(Element element) {
SetAndDecreasePriority(position_[element.Index()], element);
}
// Returns the element with given index.
Element GetElement(int index) const { return heap_[position_[index]]; }
// For i in [0, Size()) returns an element currently in the queue.
// This can be used to get a random element from the queue for instance.
Element QueueElement(int i) const { return heap_[1 + i]; }
private:
// Puts the given element at heap index i.
void Set(Element* heap, int i, Element element) {
heap[i] = element;
position_[element.Index()] = i;
}
// Puts the given element at heap index i and update the heap knowing that the
// element has a priority <= than the priority of the element currently at
// this position.
void SetAndDecreasePriority(int i, const Element element) {
const int size = size_;
Element* heap = heap_.data();
while (true) {
const int left = i * 2;
const int right = left + 1;
if (right > size) {
if (left > size) break;
const Element left_element = heap[left];
if (!less_(element, left_element)) break;
Set(heap, i, left_element);
i = left;
break;
}
const Element left_element = heap[left];
const Element right_element = heap[right];
if (less_(left_element, right_element)) {
if (!less_(element, right_element)) break;
Set(heap, i, right_element);
i = right;
} else {
if (!less_(element, left_element)) break;
Set(heap, i, left_element);
i = left;
}
}
Set(heap, i, element);
}
// Puts the given element at heap index i and update the heap knowing that the
// element has a priority >= than the priority of the element currently at
// this position.
void SetAndIncreasePriority(int i, const Element element) {
Element* heap = heap_.data();
while (i > 1) {
const int parent = i >> 1;
const Element parent_element = heap[parent];
if (!less_(parent_element, element)) break;
Set(heap, i, parent_element);
i = parent;
}
Set(heap, i, element);
}
int size_;
Compare less_;
std::vector<Element> heap_;
std::vector<int> position_;
};
} // namespace operations_research
#endif // OR_TOOLS_UTIL_INTEGER_PQ_H_