I have a const-correctness problem which I don't seem to be able to resolve. Here is the structure of my program:
class Node
{
private:
int id;
std::set<Node*> neighbours;
public:
Node();
Node(int id_p);
void set_id(const int& id_p);
int get_id() const;
void add_neighbour(Node* neighbour);
bool is_neighbour(Node* neighbour) const;
friend bool operator <(const Node& lhs, const Node& rhs);
};
class Graph
{
private:
std::set<Node> node_list;
public:
Graph();
void add_node(int id);
const Node* get_node_by_id(int id) const;
bool has_node(int id) const;
void check_add_node(int id);
void add_edge(int id_1, int id_2);
bool has_edge(int id_1, int id_2) const;
void check_add_edge(int id_1, int id_2);
(...)
};
Now the thing is, if I call the function Graph::get_node_by_id()
, I want to return a pointer to a given node (type Node
). But it seems impossible to do so, because the std::set
implicitly converts my Node type objects to const Node
objects, and I am unable fetch a non-const pointer
from a const
object.
However, I cannot have everything else set to const Node
(which would resolve the problem), because I want to call Node::add_neighbour()
from Graph::add_edge()
, but whenever I do so, my compiler says that I might be violating the const
ness (required to have a sorted set) of the elements in the node_list
set, even though I defined the less operator<
to only care about the id
.
Is there anything I can do to resolve this dilemma (without giving up on having a sorted set)? Thank you for your responses!
More info on the error:
If I use non-constant fields, error in Graph::get_node_by_id()
:
for(Node& element : this->node_list) // Error: element should be const Node&
{
if(element->get_id() == id)
{
return element;
}
}
return nullptr;
If I use constant fields, error in Graph::add_edge()
:
(...)
const Node* node_1 = this->get_node_by_id(id_1);
const Node* node_2 = this->get_node_by_id(id_2);
node_1->add_neighbour(node_2); // Error for disregarding constness
node_2->add_neighbour(node_1);
Your issue appears to be that you have two different 'value semantics' to Node
.
One is that exposed by operator<
which is not affected by add_neighbour
. This is the one set
needs, to keep things ordered, and which it enforces by making Node
const
.
The other is that exposed by the class API, where both set_id
and add_neighbour
would change the value.
To keep your sorted set
, you must not allow the id of a node to change once it's in the set. But you can allow the neighbours to change.
So I'd suggest you make the neighbours
set
mutable
, make add_neighbour
private
and const
, and make Graph
a friend
of Node
.
This is what mutable
gives you, data members that are not part of the 'value' of a type. Note that this means you are indicating that something holding a const Node*
may expect the result of is_neighbour
to change between calls.
So...
class Node
{
private:
// Trust Graph not to mess directly with these!
int id;
mutable std::set<Node*> neighbours;
friend class Graph;
// For Graph's exclusive use
void add_neighbour(Node* neighbour) const;
public:
Node();
Node(int id_p);
void set_id(const int& id_p); // Callable when not in Graph's set
int get_id() const;
void add_neighbour(Node* neighbour); // Callable when not in Graph's set
bool is_neighbour(Node* neighbour) const;
friend bool operator <(const Node& lhs, const Node& rhs);
};
class Graph
{
private:
std::set<Node> node_list;
public:
Graph();
void add_node(int id);
const Node* get_node_by_id(int id) const;
bool has_node(int id) const;
void check_add_node(int id);
void add_edge(int id_1, int id_2);
bool has_edge(int id_1, int id_2) const;
void check_add_edge(int id_1, int id_2);
(...)
};
Now what you have is public, non-const mutators for Node
instances that aren't in Graph
's set
, and an extra mutator for Graph
to use to change the neighbours of the Node
s in its set
.
So only Graph
can do
const Node b;
b.add_neighbour(nullptr);
If you really don't trust Graph
, you can replace the private
const
add_neighbour
with an inner class
, with a static add_neighbour(Node* node, Node* neighbour
method, since an inner class
is implicitly able to access private data of the outer class.
class NeighbourHelper {
friend class Graph;
static void add(const Node* node, Node* neighbour) {
node->add_neighbour(neighbour);
}
Now only Graph
can do
const Node b;
Node::NeighbourHelper::add(&b, nullptr);
In both cases, the following works for everyone:
Node a;
a.add_neighbour(nullptr);
At this point, you should be suffering a code-smell... The issue is the public
get_node_by_id
method in Graph. You actually probably want to expose an iterator of some kind instead, rather than than the raw Node*
, and make Node
a private inner class of Graph.
Or even just replace the whole Node
concept with std::map<int,std::set<int>>
...
But it depends on your actual use case.