Re: Nested class access with ->
Stephen Howe wrote:
The access rules are name-based. That means they only control which
names from a class you can access, and only the class of which the
name itself is a member is considered for the access rules.
Thanks. Where does it say that in the standard?
It can be deduced from 11/1 ([class.access]/1).
The clause explicitly refers to the name of a member that can be used,
so that makes the access rules name-based.
For private and protected access, the clause refers to the class in
which a member is declared. A member of a nested class is not
(necessarily) declared in the enclosing class, so that prevents the
inheriting of access.
I am not sure if I agree with you. On reading the intended new C++
standard, the word "name" has been stripped out ogf section 11.
Can you tell me which draft version you used? I just checked N2857 and
there section 11 is essentially the same as in my copy of the 2003
standard (the only change is the DR resolution that nested/local classes
have the same access to their enclosing class as members of that
enclosing class).
Instead "member" is used and class operators are certainly members.
And the first time "name" is mention is 11.5 - in the context of
derived and base class names.
Under the 1998 & 2003 C++ standard, your reading of it, assignment
operators would not be subject to access specifiers as they are not
names. But certainly they are subject to access specifiers, therefore
the reading is incorrect.
That depends on how strict you interpret the term 'name'. For purposes
of access control, I consider all members of a class to have one or more
names and I think that is also the intent of the standard.
Otherwise at least the constructors and the destructor, and possibly
also overloaded operators, would indeed be exempt from access control.
The reason why the name used to access a class member is relevant can be
seen in this example:
class A {
struct B {};
int i;
public:
typedef struct B BB;
int& j;
A() : i(0), j(i) {}
friend int& getI(A& that);
};
int& getI(A& that) { static int k; return rand() % 2 ? that.i : k; }
int main()
{
A a;
A::B myB; // Error: A::B is not accessible
a.i = 42; // Error: A::i is not accessible
A::BB myBB: // OK
a.j = 42; // OK
getI(a) = 24; // OK
}
Here A::B and A::BB refer to the same type, but only one of the two
names of the type is accessible.
Similarly, the two expressions 'a.i' and 'a.j' both refer to the same
object, but only one of them can be used in main to access that object.
If the access rules were not based on the name that is used to access a
member, then the example above would not be possible.
Additionally, if the access rules were based on the object being
accessed, how could a compiler determine if the expression 'getI(a) =
24' would be correct? getI() may or may not return a reference to a
private member.
It seems to me, that "->" defaults to public because I have not
defined a user-defined operator (rather like any compiler-generated
default constructor, copy constructor, destructor and assignment
operator are public by default).
You seem to have a slight misunderstanding on how operator-> works,
because there is no compiler-generated overload of that operator for
classes.
In your code, you had the situation:
T* func();
func()->member;
As func() returns a bald pointer, the only operator-> that will ever be
considered is the built-in one.
Any comments?
Thanks
Stephen Howe
Bart v Ingen Schenau
--
a.c.l.l.c-c++ FAQ: http://www.comeaucomputing.com/learn/faq
c.l.c FAQ: http://c-faq.com/
c.l.c++ FAQ: http://www.parashift.com/c++-faq-lite/
[ See http://www.gotw.ca/resources/clcm.htm for info about ]
[ comp.lang.c++.moderated. First time posters: Do this! ]