forked from google/or-tools
-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathsparse_permutation.h
160 lines (134 loc) · 5.25 KB
/
sparse_permutation.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
// Copyright 2010-2025 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.
#ifndef OR_TOOLS_ALGORITHMS_SPARSE_PERMUTATION_H_
#define OR_TOOLS_ALGORITHMS_SPARSE_PERMUTATION_H_
#include <string>
#include <vector>
#include "absl/types/span.h"
#include "ortools/base/logging.h"
namespace operations_research {
// A compact representation for permutations of {0..N-1} that displaces few
// elements: it needs only O(K) memory for a permutation that displaces
// K elements.
class SparsePermutation {
public:
explicit SparsePermutation(int size) : size_(size) {} // Identity.
// TODO(user): complete the reader API.
int Size() const { return size_; }
int NumCycles() const { return cycle_ends_.size(); }
// Returns the "support" of this permutation; that is, the set of elements
// displaced by it.
const std::vector<int>& Support() const { return cycles_; }
// The permutation has NumCycles() cycles numbered 0 .. NumCycles()-1.
// To iterate over cycle #i of the permutation, do this:
// for (const int e : permutation.Cycle(i)) { ...
struct Iterator;
Iterator Cycle(int i) const;
// This is useful for iterating over the pair {element, image}
// of a permutation:
//
// for (int c = 0; c < perm.NumCycles(); ++c) {
// int element = LastElementInCycle(c);
// for (int image : perm.Cycle(c)) {
// // The pair is (element, image).
// ...
// element = image;
// }
// }
//
// TODO(user): Provide a full iterator for this? Note that we have more
// information with the loop above. Not sure it is needed though.
int LastElementInCycle(int i) const;
// Returns the image of the given element or `element` itself if it is stable
// under the permutation.
int Image(int element) const;
int InverseImage(int element) const;
// To add a cycle to the permutation, repeatedly call AddToCurrentCycle()
// with the cycle's orbit, then call CloseCurrentCycle();
// This shouldn't be called on trivial cycles (of length 1).
void AddToCurrentCycle(int x);
void CloseCurrentCycle();
// Removes the cycles with given indices from the permutation. This
// works in O(K) for a permutation displacing K elements.
void RemoveCycles(absl::Span<const int> cycle_indices);
// Output all non-identity cycles of the permutation, sorted
// lexicographically (each cycle is described starting by its smallest
// element; and all cycles are sorted lexicographically against each other).
// This isn't efficient; use for debugging only.
// Example: "(1 4 3) (5 9) (6 8 7)".
std::string DebugString() const;
template <typename Collection>
void ApplyToDenseCollection(Collection& span) const;
private:
const int size_;
std::vector<int> cycles_;
std::vector<int> cycle_ends_;
};
inline void SparsePermutation::AddToCurrentCycle(int x) {
DCHECK_GE(x, 0);
DCHECK_LT(x, size_);
cycles_.push_back(x);
}
inline void SparsePermutation::CloseCurrentCycle() {
if (cycle_ends_.empty()) {
DCHECK_GE(cycles_.size(), 2);
} else {
DCHECK_GE(cycles_.size(), cycle_ends_.back() + 2);
}
cycle_ends_.push_back(cycles_.size());
}
struct SparsePermutation::Iterator {
// These typedefs allow this iterator to be used within testing::ElementsAre.
typedef int value_type;
typedef std::vector<int>::const_iterator const_iterator;
Iterator() = default;
Iterator(const std::vector<int>::const_iterator& b,
const std::vector<int>::const_iterator& e)
: begin_(b), end_(e) {}
std::vector<int>::const_iterator begin() const { return begin_; }
std::vector<int>::const_iterator end() const { return end_; }
const std::vector<int>::const_iterator begin_;
const std::vector<int>::const_iterator end_;
int size() const { return end_ - begin_; }
};
inline SparsePermutation::Iterator SparsePermutation::Cycle(int i) const {
DCHECK_GE(i, 0);
DCHECK_LT(i, NumCycles());
return Iterator(cycles_.begin() + (i == 0 ? 0 : cycle_ends_[i - 1]),
cycles_.begin() + cycle_ends_[i]);
}
inline int SparsePermutation::LastElementInCycle(int i) const {
DCHECK_GE(i, 0);
DCHECK_LT(i, cycle_ends_.size());
DCHECK_GT(cycle_ends_[i], i == 0 ? 0 : cycle_ends_[i - 1]);
return cycles_[cycle_ends_[i] - 1];
}
template <typename Collection>
void SparsePermutation::ApplyToDenseCollection(Collection& span) const {
using T = typename Collection::value_type;
for (int c = 0; c < NumCycles(); ++c) {
const int last_element_idx = LastElementInCycle(c);
int element = last_element_idx;
T last_element = span[element];
for (int image : Cycle(c)) {
if (image == last_element_idx) {
span[element] = last_element;
} else {
span[element] = span[image];
}
element = image;
}
}
}
} // namespace operations_research
#endif // OR_TOOLS_ALGORITHMS_SPARSE_PERMUTATION_H_