GEDLIB  1.0
bipartite.ipp
Go to the documentation of this file.
1 /***************************************************************************
2 * *
3 * Copyright (C) 2018 by David B. Blumenthal *
4 * *
5 * This file is part of GEDLIB. *
6 * *
7 * GEDLIB is free software: you can redistribute it and/or modify it *
8 * under the terms of the GNU Lesser General Public License as published *
9 * by the Free Software Foundation, either version 3 of the License, or *
10 * (at your option) any later version. *
11 * *
12 * GEDLIB is distributed in the hope that it will be useful, *
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of *
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
15 * GNU Lesser General Public License for more details. *
16 * *
17 * You should have received a copy of the GNU Lesser General Public *
18 * License along with GEDLIB. If not, see <http://www.gnu.org/licenses/>. *
19 * *
20 ***************************************************************************/
21 
27 #ifndef SRC_METHODS_BIPARTITE_IPP_
28 #define SRC_METHODS_BIPARTITE_IPP_
29 
30 namespace ged {
31 
32 template<class UserNodeLabel, class UserEdgeLabel>
33 Bipartite<UserNodeLabel, UserEdgeLabel>::
34 ~Bipartite() {}
35 
36 template<class UserNodeLabel, class UserEdgeLabel>
37 Bipartite<UserNodeLabel, UserEdgeLabel>::
38 Bipartite(const GEDData<UserNodeLabel, UserEdgeLabel> & ged_data) :
39 LSAPEBasedMethod<UserNodeLabel, UserEdgeLabel>(ged_data) {
40  this->compute_lower_bound_ = false;
41 }
42 
43 // === Definitions of member functions inherited from LSAPEBasedMethod. ===
44 template<class UserNodeLabel, class UserEdgeLabel>
45 void
47 lsape_populate_instance_(const GEDGraph & g, const GEDGraph & h, DMatrix & master_problem) {
48 
49 #ifdef _OPENMP
50  omp_set_num_threads(this->num_threads_ - 1);
51 #pragma omp parallel for if(this->num_threads_ > 1)
52 #endif
53  for (std::size_t row_in_master = 0; row_in_master < master_problem.num_rows(); row_in_master++) {
54  for (std::size_t col_in_master = 0; col_in_master < master_problem.num_cols(); col_in_master++) {
55  if ((row_in_master < g.num_nodes()) and (col_in_master < h.num_nodes())) {
56  master_problem(row_in_master, col_in_master) = compute_substitution_cost_(g, h, row_in_master, col_in_master);
57  }
58  else if (row_in_master < g.num_nodes()) {
59  master_problem(row_in_master, h.num_nodes()) = compute_deletion_cost_(g, row_in_master);
60  }
61  else if (col_in_master < h.num_nodes()) {
62  master_problem(g.num_nodes(), col_in_master) = compute_insertion_cost_(h, col_in_master);
63  }
64  }
65  }
66 }
67 
68 // === Definitions of private helper member functions. ===
69 template<class UserNodeLabel, class UserEdgeLabel>
70 double
73  // Collect node substitution costs.
74  double cost{this->ged_data_.node_cost(g.get_node_label(i), h.get_node_label(k))};
75 
76  // Initialize subproblem.
77  DMatrix subproblem(g.degree(i) + 1, h.degree(k) + 1);
78 
79  // Collect edge deletion costs.
80  std::size_t j{0};
81  for (auto ij = g.incident_edges(i).first; ij != g.incident_edges(i).second; ij++, j++) {
82  subproblem(j, h.degree(k)) = this->ged_data_.edge_cost(g.get_edge_label(*ij), ged::dummy_label());
83  }
84 
85  // Collect edge insertion costs.
86  std::size_t l{0};
87  for (auto kl = h.incident_edges(k).first; kl != h.incident_edges(k).second; kl++, l++) {
88  subproblem(g.degree(i), l) = this->ged_data_.edge_cost(ged::dummy_label(), h.get_edge_label(*kl));
89  }
90  j = 0;
91 
92  // Collect edge relabelling costs.
93  for (auto ij = g.incident_edges(i).first; ij != g.incident_edges(i).second; ij++, j++) {
94  l = 0;
95  for (auto kl = h.incident_edges(k).first; kl != h.incident_edges(k).second; kl++, l++) {
96  subproblem(j, l) = this->ged_data_.edge_cost(g.get_edge_label(*ij), h.get_edge_label(*kl));
97  }
98  }
99 
100  // Solve subproblem.
101  LSAPESolver subproblem_solver(&subproblem);
102  subproblem_solver.set_model(this->lsape_model_);
103  subproblem_solver.solve();
104 
105  // Update and return overall substitution cost.
106  cost += subproblem_solver.minimal_cost();
107  return cost;
108 }
109 
110 template<class UserNodeLabel, class UserEdgeLabel>
111 double
114  // Collect node deletion cost.
115  double cost{this->ged_data_.node_cost(g.get_node_label(i), ged::dummy_label())};
116 
117  // Collect edge deletion costs.
118  auto incident_edges_i = g.incident_edges(i);
119  for (auto ij = incident_edges_i.first; ij != incident_edges_i.second; ij++) {
120  cost += this->ged_data_.edge_cost(g.get_edge_label(*ij), ged::dummy_label());
121  }
122 
123  // Return overall deletion cost.
124  return cost;
125 }
126 
127 template<class UserNodeLabel, class UserEdgeLabel>
128 double
131  // Collect node insertion cost.
132  double cost{this->ged_data_.node_cost(ged::dummy_label(), h.get_node_label(k))};
133 
134  // Collect edge insertion costs.
135  auto incident_edges_k = h.incident_edges(k);
136  for (auto kl = incident_edges_k.first; kl != incident_edges_k.second; kl++) {
137  cost += this->ged_data_.edge_cost(ged::dummy_label(), h.get_edge_label(*kl));
138  }
139 
140  // Return overall insertion cost.
141  return cost;
142 }
143 
144 }
145 
146 #endif /* SRC_METHODS_BIPARTITE_HPP_ */
void set_model(const Model &model)
Makes the solver use a specific model for optimal solving.
std::size_t num_nodes() const
Returns the number of nodes.
Definition: ged_graph.ipp:211
std::size_t num_threads_
The number of threads to be used.
This class solves LSAPE instances by calling the library lsape available at https://bougleux.users.greyc.fr/lsape/.
virtual void lsape_populate_instance_(const GEDGraph &g, const GEDGraph &h, DMatrix &lsape_instance) final
Populates the LSAPE instance.
Definition: bipartite.ipp:47
const GEDData< UserNodeLabel, UserEdgeLabel > & ged_data_
The data on which the method is run.
Definition: ged_method.hpp:124
void solve(int num_solutions=1)
Solves the LSAPE problem instance.
std::size_t degree(NodeID node) const
Returns node degree.
Definition: ged_graph.ipp:162
std::size_t num_cols() const
Returns the number of columns.
Definition: matrix.ipp:110
LabelID get_node_label(NodeID node) const
Returns the label of a given node.
Definition: ged_graph.ipp:126
bool compute_lower_bound_
Flag that should be set to true if and only if the method computes a lower bound. ...
Computes lower and upper bounds for general edit costs.
Definition: bipartite.hpp:42
LSAPESolver::Model lsape_model_
Specifies model for optimal LSAPE solver.
LabelID get_edge_label(EdgeID edge) const
Returns the label of a given edge.
Definition: ged_graph.ipp:135
std::pair< incident_edge_iterator, incident_edge_iterator > incident_edges(NodeID node) const
Provides access to all incident edges of a node.
Definition: ged_graph.ipp:150
double minimal_cost() const
Returns the cost of the computed solutions.
The normalized input graphs used by GEDLIB. All labels are integers.
Definition: ged_graph.hpp:104
constexpr LabelID dummy_label()
Returns a dummy label.
Global namespace for GEDLIB.
std::size_t NodeID
Internally used vertex ID type.
Definition: ged_graph.hpp:108
std::size_t num_rows() const
Returns the number of rows.
Definition: matrix.ipp:85