[comp.lang.c++] const friend functions

ned@pebbles.cad.mcc.com (Ned Nowotny) (03/05/90)

In article <6553@cadillac.CAD.MCC.COM> ned@MCC.COM (Ned Nowotny) writes:
> ...
> (For example, consider const friend functions:  Everything I have read,
> including Lippman and the Release 2.0 Reference manual, SEEM to indicate
> that only member functions may be declared const functions. ...

Now that I am no doubt too late to prevent a hoard of follow ups and replies
which will completely drown out the discussion pertaining to the worthiness
of being able to specify C++ syntax with a LALR(1) grammar, I am compelled to
confess that I do indeed realize that only member functions may be declared
const.  For those who have not instinctively determined this for themselves,
a brief intuitive explanation will follow.  For most everyone else, I would
just request that you consider that the Cfront 2.0 behavior should be
considered sufficient evidence that semantic processing in "smart" lexical
analyzers is not a good substitute for a syntax that can be precisely
encapsulated within the expression of a given context-free grammar.  After
all, the issue here pertains to a programming language specification and not
to the inherent problems and requirements in processing natural languages.

In any case, on with what will undoubtably be yet another flawed explanation
to be clarified by language lawyers far more versed in C++ than myself:

For those of you who have not noticed already, C++ requires complete
function prototypes in all but two cases.  The first is where the
programmer elides the remaining formal arguments to a function with
the explicit use of ellipses (...).  However, this case is not of
immediate concern.  The other case is that of member functions.

In member function declarations, the first formal argument is omitted
as a notational convenience.  The first argument is, of course, "this."
Unfortunately, this notational convenience reveals a darker inconvenience
in two cases which are addressed in C++ through some rather unfortunate
mechanisms.  (However, I do not have an alternative to offer.  Then again,
the current mechanisms are unlikely to change.)

One problem with the current system partially explains the need for
friend functions to implement commutative operator functions.  Since
the first argument of a member function is always the "this" pointer,
binary operator functions which attempt to implement the commutative
property must be friend functions so that the first argument may be
of a different class type than the second argument which is of the class
type which would allow a member function implementation for the commutative
operation.  (My apologies for that rather opaque description.  I can only
hope that it is reasonably precise.)  Of course, the ability to declare
member functions with the "this" pointer explicitly given a different
position in the formal argument list from the currently implicit first
position would not necessarily be sufficient to support commutative
operator member functions, but it might help.

The other problem with a hidden formal "this" argument (and the one
which finally bears directly on the subject at hand), is that the
"this" pointer's type declaration is also hidden.  As a result, there
was no mechanism to add the "const" or "volatile" type qualifiers
when needed.  As a result, it was apparently decided that member
functions would declare that the object pointed to by the "this"
argument was "const" (or, I assume, "volatile" or "const volatile" ...)
by way of the const member function syntax.  For example, the following
two declarations are equivalent (although the one with the explicit
"this" argument is obviously not the same if you assume that it also
includes the implicit "this" argument as well):

// Current method
class A
{
    int Value;

    public:
	int GetValue() const { return Value; }
};

// Hypothetical explicit "this" declaration syntax
class B
{
    int Value;

    public:
	int GetValue( const B * const this ) { return this->Value; }
};

Since the "const" and "volatile" type qualifiers can be explicitly used
in all other function prototypes and since a "const" friend function
would not necessarily allow the compiler to determine to which formal argument
the "const" qualifier was to be bound, const friend functions are not only
unnecessary, they are also potentially ambiguous.

Disclaimer:  None of the preceding should be considered entirely accurate.
	     There are obviously more qualified people than myself to
	     clear up any questions about this language feature.  I only
	     thought it best to descibe my own best understanding of the
	     feature after having brought it up in the first place.

Ned Nowotny, MCC CAD Program, Box 200195, Austin, TX  78720  Ph: (512) 338-3715
ARPA: ned@mcc.com                   UUCP: ...!cs.utexas.edu!milano!cadillac!ned
-------------------------------------------------------------------------------
"We have ways to make you scream." - Intel advertisement in the June 1989 DDJ.