Say that a base class A
defines a protected member. A derived class B
uses this member.
class A
{
public:
A(int v) : value(v) { }
protected:
int value;
};
class B : public A
{
public:
B(int v) : A(v) { }
void print() const;
void compare_and_print(const A& other) const;
};
The function B::print
just takes the value of the current member and prints it:
void B::print() const
{
std::cout << "Value: " << value << "\n";
}
The other member function, B::compare_and_print
, takes an instance of A
, checks their values and prints the maximum of both:
void B::compare_and_print(const A& other) const
{
auto max_value = std::max(value, other.value);
std::cout << "Max value: " << max_value << "\n";
}
If other
were an instance of the class B
, this would be no problem. But the function would like to work with any kind of A
instances. This does unfortunately not compile. This is what clang says about this:
test.cpp:27:42: error: 'value' is a protected member of 'A'
auto max_value = std::max(value, other.value);
^
test.cpp:9:7: note: can only access this member on an object of type 'B'
int value;
^
1 error generated.
This restriction sounds counter-intuitive to me. However, I'm not going to dispute the C++ standard about this (I'm nevertheless interested about the rationale behind this decision).
My problem is that in my project I really have this kind of use case: A derived class has a method that takes an instance of the base class and needs to access a protected member of the received object.
The easiest solution, the one I implemented currently, is to add a public member function to the base class, which returns the protected member. This solution does not fully satisfy me because I would like to avoid exporting the member this way.
How can I enable the usage of the base class' protected member by the derived class without exporting the member through the API?
EDIT: Some related questions are:
The answer I would like to have is the explanation of a design pattern to solve this problem without exposing the the protected member to the external code (where external means, code that is not part of the framework defining these classes).
It could be that such a pattern cannot exist, I acknowledge.
There is actually a loophole using member pointers (no casting, no copying):
You can bypass protected by using a helper struct:
You can get it by using it anywhere
A_Helper::GetProtectedValue(a)
In your case, you could cast
other
toconst B&
(viastatic_cast
orreinterpret_cast
) but you don't know if instance ofother
is of typeB
. With that casted value people reading the code would presume thatother
is of typeB
and could insert code that causes reads/writes to "random" memory.Consider that class
B
has another memebervalue_B
andother
is of typeC
. Usingstatic_cast<const B&>(other).value_B
is Undefined Behavior.