A derived class can call a protected base class constructor in its ctor-initializer, but only for its own base class subobject, and not elsewhere:
class Base {
protected:
Base() {}
};
class Derived : Base {
Base b;
public:
Derived(): Base(), // OK
b() { // error
Base b2; // error
}
};
What does the standard say about this? Here is [class.protected]/1:
An additional access check beyond those described earlier in Clause 11 is applied when a non-static data member or non-static member function is a protected member of its naming class (11.2) As described earlier, access to a protected member is granted because the reference occurs in a friend or member of some class
C
. If the access is to form a pointer to member (5.3.1), the nested-name-specifier shall denoteC
or a class derived fromC
. All other accesses involve a (possibly implicit) object expression (5.2.5). In this case, the class of the object expression shall beC
or a class derived fromC
. [ Example: ...
Is there an object expression involved when calling a constructor? There isn't, is there? So where in the standard is access control for protected base class constructors described?
The protected
access only applies to parent members of your own current object type. You don't get public access to the protected members of other objects of the parent type. In your example you only get access to the default base contructor as part of a Derived
, not when it's a standalone object as b
.
Let's break down the quote you posted from the standard (11.4/1). We'll assume that C
in the standard corresponds to your Derived
class:
An additional access check beyond those described earlier in Clause 11 is applied when a non-static data member or non-static member function is a protected member of its naming class (11.2).
So the base class constructor is effectively a non-static member function
of its naming class (B
) here, so this clause applies so far.
As described earlier, access to a protected member is granted because the reference occurs in a friend or member of some class C.
Member (constructor) of C
so we're still good here.
If the access is to form a pointer to member (5.3.1), the nested-name-specifier shall denote C or a class derived from C.
This is not a pointer to member so this doesn't apply.
All other accesses involve a (possibly implicit) object expression (5.2.5).
The standard then asserts that all other possible accesses must involve an object expression.
In this case, the class of the object expression shall be C or a class derived from C.
Finally the standard states that the class of the expression must be C
or a further derived class. In this case your expression Base()
is in fact a C
, calling the parent constructor (think about it as this->Base()
. The expression b
is clearly of type Base
(that's the explicitly declared type of the member b
, think about this->b->Base()
). Now we do the check: Is Base
a C
or child of C
? It is not, so the code is not legal.
C++11 §11.2/5:
”
A memberm
is accessible at the point R when named in classN
if
m
as a member ofN
is public, or
m
as a member ofN
is private, and R occurs in a member or friend of classN
, or
m
as a member ofN
is protected, and R occurs in a member or friend of classN
, or in a member or friend of a classP
derived fromN
, wherem
as a member ofP
is public, private, or protected, orthere exists a base class
B
ofN
that is accessible at R, andm
is accessible at R when named in classB
.
For your constructor invocation
Base b2;
the 3rd point above applies. m
is the Base
constructor. N
, the naming class, is Base
. m
as a member of Base
is protected, and the declaration occurs in a member of class Derived
derived from Base
, but it's not the case that the Base
constructor as a member of Derived
is public, private or protected: it’s simply not a member of Derived
, constructors are not implicitly inherited.
I think the language “is public, private, or protected” is pretty awkward; I can only surmise that it’s the result of some evolution of this paragraph.
I have yet to find an explanation of how formally the protected Base
constructor is accessible in a member initializer list in Derived
, but then I just started looking at this for this question.
Update: I fail to find any language in the standard pertaining to access to constructors in an initializer list, and I fail to find any Defect Report about it. It’s quite possibly a defect.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With