22#include "absl/container/flat_hash_map.h"
23#include "absl/types/span.h"
27#include "ortools/math_opt/sparse_containers.pb.h"
36 std::optional<int64_t>
end;
39 bool Contains(
const int64_t
id)
const {
40 return id >=
start && (!
end.has_value() ||
id < *
end);
47 const SparseDoubleMatrixProto& matrix,
const int64_t start_row_id,
48 const std::optional<int64_t> end_row_id,
const int64_t start_col_id,
49 const std::optional<int64_t> end_col_id) {
50 const int matrix_size = matrix.row_ids_size();
51 CHECK_EQ(matrix_size, matrix.column_ids_size());
52 CHECK_EQ(matrix_size, matrix.coefficients_size());
53 const IndexRange row_range = {.start = start_row_id, .end = end_row_id};
54 const IndexRange col_range = {.start = start_col_id, .end = end_col_id};
59 for (
int row_start = 0, next_row_start; row_start < matrix_size;
62 row_start = next_row_start) {
65 const int64_t row_id = matrix.row_ids(row_start);
66 int row_end = row_start + 1;
67 while (row_end < matrix_size && matrix.row_ids(row_end) == row_id) {
72 next_row_start = row_end;
75 if (!row_range.Contains(row_id)) {
80 int row_cols_start = row_start;
81 while (row_cols_start < row_end &&
82 !col_range.Contains(matrix.column_ids(row_cols_start))) {
88 int row_cols_end = row_cols_start;
89 while (row_cols_end < row_end &&
90 col_range.Contains(matrix.column_ids(row_cols_end))) {
93 const int row_cols_len = row_cols_end - row_cols_start;
95 if (row_cols_len != 0) {
96 filtered_rows.emplace_back(
97 row_id,
MakeView(absl::MakeConstSpan(matrix.column_ids())
98 .subspan(row_cols_start, row_cols_len),
99 absl::MakeConstSpan(matrix.coefficients())
100 .subspan(row_cols_start, row_cols_len)));
104 return filtered_rows;
111 absl::flat_hash_map<int64_t, SparseVector<double>> filtered_columns;
112 for (
const auto& [row_id, column_values] : submatrix_by_rows) {
113 for (
const auto [column_id,
value] : column_values) {
115 row_values.
ids.push_back(row_id);
121 std::vector<std::pair<int64_t, SparseVector<double>>> sorted_filtered_columns(
122 std::make_move_iterator(filtered_columns.begin()),
123 std::make_move_iterator(filtered_columns.end()));
124 std::sort(sorted_filtered_columns.begin(), sorted_filtered_columns.end(),
127 return lhs.first < rhs.first;
130 return sorted_filtered_columns;
#define CHECK_EQ(val1, val2)
std::vector< std::pair< int64_t, SparseVector< double > > > TransposeSparseSubmatrix(const SparseSubmatrixRowsView &submatrix_by_rows)
std::vector< std::pair< int64_t, SparseVectorView< double > > > SparseSubmatrixRowsView
SparseSubmatrixRowsView SparseSubmatrixByRows(const SparseDoubleMatrixProto &matrix, const int64_t start_row_id, const std::optional< int64_t > end_row_id, const int64_t start_col_id, const std::optional< int64_t > end_col_id)
SparseVectorView< T > MakeView(absl::Span< const int64_t > ids, const Collection &values)
std::optional< int64_t > end
std::vector< int64_t > ids