13Clusters::hasAllZeros()
const
15 return !c.empty() && c[0] == 0 && c.size() == 1 && c_ind.empty();
19Clusters::getZeroIndices()
const
21 if (!zero_indices_valid) {
25 for (
int j = 0; j < p; ++j) {
26 if (std::find(c_ind.begin(), c_ind.end(), j) == c_ind.end()) {
27 zero_indices.push_back(j);
31 zero_indices_valid =
true;
37std::vector<int>::const_iterator
40 if (i <
static_cast<int>(c.size())) {
41 return c_ind.cbegin() + c_ptr[i];
44 return getZeroIndices().cbegin();
47std::vector<int>::const_iterator
50 if (i <
static_cast<int>(c.size())) {
51 return c_ind.cbegin() + c_ptr[i + 1];
54 return getZeroIndices().cend();
57std::vector<int>::iterator
60 if (i <
static_cast<int>(c.size())) {
61 return c_ind.begin() + c_ptr[i];
64 return getZeroIndices().begin();
67std::vector<int>::iterator
70 if (i <
static_cast<int>(c.size())) {
71 return c_ind.begin() + c_ptr[i + 1];
74 return getZeroIndices().end();
80 if (i <
static_cast<int>(c.size())) {
81 return c_ptr[i + 1] - c_ptr[i];
85 if (i ==
static_cast<int>(c.size()) && p >
static_cast<int>(c_ind.size())) {
86 return p -
static_cast<int>(c_ind.size());
95 if (i <
static_cast<int>(c_ptr.size())) {
113 return c.size() + (p >
static_cast<int>(c_ind.size()) ? 1 : 0);
119 if (i <
static_cast<int>(c.size())) {
135 std::vector<double> result = c;
144 if (p >
static_cast<int>(c_ind.size())) {
145 result.push_back(0.0);
167 zero_indices_valid =
false;
169 auto c_old =
coeff(old_index);
171 if (c_new != c_old) {
172 if (c_new ==
coeff(new_index)) {
173 merge(old_index, new_index);
176 if (old_index != new_index) {
177 reorder(old_index, new_index);
186 using sort_pair = std::pair<double, int>;
196 zero_indices_valid =
false;
198 std::vector<sort_pair> sorted;
202 for (
int i = 0; i < beta.size(); ++i) {
203 signs.emplace_back(
sign(beta(i)));
204 double abs_val = std::abs(beta(i));
206 sorted.emplace_back(abs_val, i);
211 if (sorted.empty() && p > 0) {
219 std::sort(sorted.begin(), sorted.end(), std::greater<sort_pair>());
221 c_ind.reserve(sorted.size());
222 for (
const auto& sorted_i : sorted) {
223 c_ind.emplace_back(sorted_i.second);
227 std::vector<sort_pair> sorted_unique;
228 sorted_unique.reserve(sorted.size());
230 for (
auto it = sorted.begin(); it != sorted.end();) {
231 const double current_value = it->first;
232 sorted_unique.emplace_back(*it);
233 it = std::find_if(it, sorted.end(), [current_value](
const sort_pair& elem) {
234 return elem.first != current_value;
238 c.reserve(sorted_unique.size());
239 for (
const auto& sorted_unique_i : sorted_unique) {
240 c.emplace_back(sorted_unique_i.first);
243 c_ptr.reserve(c.size() + 1);
244 c_ptr.emplace_back(0);
246 auto range_start = sorted.begin();
247 for (
const auto& c_i : c) {
249 std::find_if(range_start, sorted.end(), [&c_i](
const sort_pair& x) {
250 return x.first != c_i;
252 c_ptr.emplace_back(std::distance(sorted.begin(), range_end));
253 range_start = range_end;
258Clusters::reorder(
const int old_index,
const int new_index)
269 if (new_index < old_index) {
272 std::for_each(c_ptr.begin() + new_index + 1,
273 c_ptr.begin() + old_index + 2,
274 [c_size](
int& x) { x += c_size; });
276 c_ptr[new_index + 1] = c_ptr[new_index] + c_size;
280 std::for_each(c_ptr.begin() + old_index,
281 c_ptr.begin() + new_index,
282 [c_size](
int& x) { x -= c_size; });
283 c_ptr[new_index] = c_ptr[new_index + 1] - c_size;
288Clusters::merge(
const int old_index,
const int new_index)
293 c.erase(c.cbegin() + old_index);
299 if (new_index < old_index) {
300 std::for_each(c_ptr.begin() + new_index + 1,
301 c_ptr.begin() + old_index + 1,
302 [c_size](
int& x) { x += c_size; });
304 std::for_each(c_ptr.begin() + old_index + 1,
305 c_ptr.begin() + new_index + 1,
306 [c_size](
int& x) { x -= c_size; });
309 c_ptr.erase(c_ptr.begin() + old_index + 1);
312std::vector<std::vector<int>>
315 std::vector<std::vector<int>> clusters;
320 std::vector<int> zero_cluster;
321 for (
int i = 0; i < p; ++i) {
322 zero_cluster.push_back(i);
324 clusters.push_back(zero_cluster);
335Eigen::SparseMatrix<int>
338 std::vector<Eigen::Triplet<int>> triplets;
351 for (
auto it =
cbegin(k); it !=
cend(k); ++it) {
354 triplets.emplace_back(ind, k, s);
358 Eigen::SparseMatrix<int> out(p, n_cols);
359 out.setFromTriplets(triplets.begin(), triplets.end());
364Eigen::SparseMatrix<int>
Representation of the clusters in SLOPE.
void update(const int old_index, const int new_index, const double c_new)
Updates the cluster structure when an index is changed.
double coeff(const int i) const
Returns the coefficient of the cluster with the given index.
Eigen::SparseMatrix< int > patternMatrix() const
Returns the cluster pattern as a sparse matrix.
int cluster_size(const int i) const
Returns the size of the cluster with the given index.
int pointer(const int i) const
Returns the pointer of the cluster with the given index.
std::vector< std::vector< int > > getClusters() const
Returns the clusters as a vector of vectors.
std::vector< int > indices() const
Returns a vector containing the indices of all clusters.
void setCoeff(const int i, const double x)
Sets the coefficient of the cluster with the given index.
int n_clusters() const
Returns the number of clusters.
std::vector< int >::iterator begin(const int i)
Returns an iterator pointing to the beginning of the cluster with the given index.
Clusters()=default
Constructs an Clusters object.
std::vector< int >::const_iterator cend(const int i) const
Returns a constant iterator pointing to the end of the cluster with the given index.
std::vector< int > pointers() const
Returns a vector containing the pointers of all clusters.
std::vector< int >::iterator end(const int i)
Returns an iterator pointing to the end of the cluster with the given index.
std::vector< double > coeffs() const
Returns a vector containing the coefficients of all clusters.
std::vector< int >::const_iterator cbegin(const int i) const
Returns a constant iterator pointing to the beginning of the cluster with the given index.
The declaration of the Clusters class.
Mathematical support functions for the slope package.
Namespace containing SLOPE regression implementation.
void move_elements(std::vector< T > &v, const int from, const int to, const int size)
int sign(T val)
Returns the sign of a given value.
Eigen::SparseMatrix< int > patternMatrix(const Eigen::VectorXd &beta)
Returns the cluster pattern as a sparse matrix.
Various utility functions.