[comp.lang.c++] More nits to pick at...

rfg@riunite.ACA.MCC.COM (Ron Guilmette) (05/01/89)

Listen friends... Is it just me or is this language difficult to
understand and/or not rigidly specified at this time?

(Actually, I already know the answer.  That is just my way of
trying to excuse myself for pestering you all with all of my
nit picking questions.  I hope that I will be forgiven if I
occasionally break into the middle of these Socratic discussions
of the true meaning of "polymorphism" to ask some rather more
mundane questions about C++ which the poor schmucks down in the
trenches {like me} may need to know about.)

So here is edition #2 of my nit picking questions.  I hope that
even the language lawyers will find at least one brain teaser
in here.

As before, I would welcome *any* responses from *anybody* who feels
that they may have a handle on how things really are (or better yet
how they should be, or how they will be when "2.0: The Movie"  arrives
for your Summer (Ahem... Fall? ... Winter?) viewing pleasure.) :-)

-----------------------------------------------------------------------
How can you declare a function which can accept a pointer
to itself as an argument (in a type-safe manner of course)?

You can now inherit a base class as either public or private.
Can you inherit a base class as "protected"?  If so, what does
this mean?

Is it legal to declare an operator which takes no
struct/class parameters but which does take at least
one union parameter?  At least one enum parameter?

Is it an error if there is either an implicit or an explicit
call to a constructor (or a destructor) for a given class at
a point where the given constructor (or destructor) is not
"visible" because it is either private or protected?

If a constructor makes calls to virtual member functions
which are declared (and defined) for the constructor's
own class, do these calls ever go through the virtual
function table?  If so why?

Is there any way to specify (explicitly) the order in which
members and base classes are destructed?

Is it legal to declare a function as inline and later give
either another declaration or a definition for the function
whose heading does *not* include the keyword "inline"?

Is it legal to declare a function as non-inline and later
to redeclare it (or give a definition for it) which is inline?

Is it legal for a member function to be both inline and virtual?
If so, should the "inline" keyword come before or after the
"virtual" keyword?

Are enum type values always treated like int type values
or can they be used to disambiguate a function call, i.e.:

	enum color { red, yellow };
	enum fruit { banana, apple, orange };

	void overloaded (enum color);
	void overloaded (enum fruit);
	void overloaded (int i);

		...
		overloaded (red);
		overloaded (apple);
		overloaded (99);
		...

At the end of section 8.9 in the reference manual there is a short
note about how to get the address of a particular instance of an
overloaded function. It says:

	"The address-of-operator & may only be applied to an overloaded
	name in an assignment or initialization where the type expected
	determines which function to take the address of."

Here, the term "the type expected" is not really defined.  It is thus, not
clear (as it is in the definition of the Ada language) what elements of
the context may (or may not) be considered when a "standard conforming"
compiler attempts to perform disambiguation.  Specifically, I would like
to know if the following should be considered legal (i.e. unambiguous):

	class A { ... };
	class B { ... };

	void overloaded (A);
	void overloaded (B);

	void* pointer = (void (*)(A)) &overloaded;

What are "static member functions"?  I seem to recall seeing something
which related this term somehow to "operator new" and "operator delete"
members.  Are these two (special?) operator members implicitly static?

Do "static" member functions (and operators) lack a "this"
pointer?  If so, is it an error to refer to "this" within the body of
such operators.

May a default argument be a non-constant expression?

May an expression supplied as a default function argument contain a reference to:

	a) the "this" pointer (for member functions), or
	b) earlier parameters in the same parameter list, or
	c) later parameters in the same parameter list, or
	d) the address of the function for which the formal parameter
	   list is being specified?

Are default argument expressions compile-time evaluated in the context
of the default argument expressions themselves, or are they compile-time
evaluated in the context of each individual call in which they are used?
For instance, will the following print 3 or 7?

	const int k = 3;

	int f (int i = k)
	{
		printf ("%d\n", i);
	}

	int main ()
	{
		const int k = 7;

		f ();
	}

Is it legal or illegal to put an "overload" statement inside a function?
Inside a class?  A struct?  A union?  Is it ever of any value to nest
overload statements within these contexts?

Is the following legal?

	overload x;
	overload x;
	overload x;

Is the following legal?  Is it ever required?

	overload operator+;

Is the following legal?

	class base {
	public:
		overload member;
		void member ();
		void member (int i);
	};

Is the following legal?

	int global_item;
	overload global_item;
	void global_item (int i);

Is there any possible ordering of the three statements above which would be
considered legal?

Is it ever legal to specify a return type for either a constructor or
a destructor?  Specifically, is it legal for constructors and/or
destructors to return (a) void, or (b) void*, or (c) class_type*,
where "class_type" is the name of the containing class?

Is it legal or illegal to place a declaration for a member function
within one "visibility section" (i.e. "public", "private", or "protected")
of a class declaration and then to subsequently re-declare the same
member function (with the same parameter list) in a different "visibility
section" of the same class declaration, for example:

	class base {
	protected:
		void member (int i);
	public:
		void member (int i);		// legal ?
	private:
		void member (int i) {}		// legal ??
	};
-------------------------------------------------------------------------------

-- 
// Ron Guilmette  -  MCC  -  Experimental Systems Kit Project
// 3500 West Balcones Center Drive,  Austin, TX  78759  -  (512)338-3740
// ARPA: rfg@mcc.com
// UUCP: {rutgers,uunet,gatech,ames,pyramid}!cs.utexas.edu!pp!rfg

ark@alice.UUCP (Andrew Koenig) (05/02/89)

Here's my try at answering Ron's questions.  My comments are indented:
------------------------------------------------------------------------

Listen friends... Is it just me or is this language difficult to
understand and/or not rigidly specified at this time?

	A major factor in how easy most languages are to
	understand is how many of your immediate colleagues
	are using it.  If a lot of them are, you can walk
	across the hall and ask for help if you get stuck.
	People had plenty of trouble understanding C while
	it was getting started.

	C++ is now about as far along as C was in 1979.
	The reference manual now in the works will go a
	long way toward a more solid specification.

(Actually, I already know the answer.  That is just my way of
trying to excuse myself for pestering you all with all of my
nit picking questions.  I hope that I will be forgiven if I
occasionally break into the middle of these Socratic discussions
of the true meaning of "polymorphism" to ask some rather more
mundane questions about C++ which the poor schmucks down in the
trenches {like me} may need to know about.)

	Fair enough.

	The answers that follow are all based on C++ 2.0.
	For that matter, several of your questions seem to
	refer to things that are present only in C++ 2.0.

	Bjarne intends to make the reference manual generally
	available when it's ready, along with a (long) paper
	describing all the stuff in 2.0 that's not in the book.

So here is edition #2 of my nit picking questions.  I hope that
even the language lawyers will find at least one brain teaser
in here.

As before, I would welcome *any* responses from *anybody* who feels
that they may have a handle on how things really are (or better yet
how they should be, or how they will be when "2.0: The Movie"  arrives
for your Summer (Ahem... Fall? ... Winter?) viewing pleasure.) :-)

	Summer.  We're still on target for June 30.

-----------------------------------------------------------------------
How can you declare a function which can accept a pointer
to itself as an argument (in a type-safe manner of course)?

	You can't in C++ or C.

You can now inherit a base class as either public or private.
Can you inherit a base class as "protected"?  If so, what does
this mean?

	You can't inherit a base class as protected.
	We need more experience with the present protection
	semantics before introducing new notions.
	There are plenty of new notions possible -- the
	hard part is picking the right ones.

Is it legal to declare an operator which takes no
struct/class parameters but which does take at least
one union parameter?  At least one enum parameter?

	A union is a class, so yes. An enumeration isn't a class, so no.

Is it an error if there is either an implicit or an explicit
call to a constructor (or a destructor) for a given class at
a point where the given constructor (or destructor) is not
"visible" because it is either private or protected?

	Yes.

If a constructor makes calls to virtual member functions
which are declared (and defined) for the constructor's
own class, do these calls ever go through the virtual
function table?  If so why?

	Suppose that class D is derived from B.
	Then constructing a D involves constructing a B first.

	Objects are constructed in exactly the same way
	regardless of whether they are part of other objects.
	Thus while the B is being constructed, it has no
	knowledge of whether or not it's part of a D.

	It follows from this that if B's constructor calls
	one of B's virtual functions, directly or indirectly,
	the behavior is as if the B were being constructed
	in isolation. You get the function from B or
	one of its base classes, but never the function
	from D or any other derived class of B.

Is there any way to specify (explicitly) the order in which
members and base classes are destructed?

	They are destroyed in inverse order from construction.
	There is no way to control the order of destruction
	except by ordering your declarations to control the
	order of construction.  Why would you want to?

Is it legal to declare a function as inline and later give
either another declaration or a definition for the function
whose heading does *not* include the keyword "inline"?

	Yes.

Is it legal to declare a function as non-inline and later
to redeclare it (or give a definition for it) which is inline?

	Yes, but if you want to declare a function inline,
	you must do so before the first place you use it.

Is it legal for a member function to be both inline and virtual?
If so, should the "inline" keyword come before or after the
"virtual" keyword?

	Yes it is legal.  However, it is never necessary to
	say `inline' and `virtual' together.

	Example 1:

		struct X {
			virtual void f() { /* ... */ }
		};

	Here, X::f is inline by virtue of being defined
	(not just declared) inside a class definition.
	There is therefore no need to say `inline' (but it's
	legal, either before or after `virtual').

	Example 2:

		struct X {
			virtual void f();
		};

		inline void X::f() { /* ... */ }

	Here the declaration and definition are separate, so
	there's still no need to say `inline' and `virtual' together.

	Although it's never necessary, it's generally legal;
	in such contexts, they may appear in either order.

Are enum type values always treated like int type values
or can they be used to disambiguate a function call, i.e.:

	enum color { red, yellow };
	enum fruit { banana, apple, orange };

	void overloaded (enum color);
	void overloaded (enum fruit);
	void overloaded (int i);

		...
		overloaded (red);
		overloaded (apple);
		overloaded (99);
		...

	Enumerated types are separate types, so they can
	be used to disambiguate a function call.  Your example
	above is legal.  The following, though, is not:

		enum color { red, yellow, orange };
		enum fruit { apple, orange, banana };

	What would `orange' represent?  What type would it be?

At the end of section 8.9 in the reference manual there is a short
note about how to get the address of a particular instance of an
overloaded function. It says:

	"The address-of-operator & may only be applied to an overloaded
	name in an assignment or initialization where the type expected
	determines which function to take the address of."

Here, the term "the type expected" is not really defined.  It is thus, not
clear (as it is in the definition of the Ada language) what elements of
the context may (or may not) be considered when a "standard conforming"
compiler attempts to perform disambiguation.  Specifically, I would like
to know if the following should be considered legal (i.e. unambiguous):

	class A { ... };
	class B { ... };

	void overloaded (A);
	void overloaded (B);

	void* pointer = (void (*)(A)) &overloaded;

	Sorry.  The only places you can do overloaded selection is
	in assignment and initialization.  However, this is OK:

		void (*p)(A) = overloaded;
		void* pointer = p;

	Note, though, that some underlying C implementations may
	prohibit converting a function pointer to a void*.

What are "static member functions"?  I seem to recall seeing something
which related this term somehow to "operator new" and "operator delete"
members.  Are these two (special?) operator members implicitly static?

	Yes.

Do "static" member functions (and operators) lack a "this"
pointer?  If so, is it an error to refer to "this" within the body of
such operators.

	Yes to both questions.

May a default argument be a non-constant expression?

	Yes.

May an expression supplied as a default function argument contain a reference to:

	a) the "this" pointer (for member functions), or

		No.

	b) earlier parameters in the same parameter list, or

		No.

	c) later parameters in the same parameter list, or

		No.

	d) the address of the function for which the formal parameter
	   list is being specified?

		If this means what I think it does, the answer is no --
		how would you write the type of the argument?

Are default argument expressions compile-time evaluated in the context
of the default argument expressions themselves, or are they compile-time
evaluated in the context of each individual call in which they are used?
For instance, will the following print 3 or 7?

	const int k = 3;

	int f (int i = k)
	{
		printf ("%d\n", i);
	}

	int main ()
	{
		const int k = 7;

		f ();
	}

	They're evaluated at run time at the point of call, but in
	the context in which they were written.  In this example,
	f() is equivalent to f(3).

Is it legal or illegal to put an "overload" statement inside a function?
Inside a class?  A struct?  A union?  Is it ever of any value to nest
overload statements within these contexts?

	The `overload' declaration is obsolescent in 2.0 --
	all functions are (potentially) overloaded.

Is the following legal?

	overload x;
	overload x;
	overload x;

		Yes.

Is the following legal?  Is it ever required?

	overload operator+;

		Yes and no, respectively.

Is the following legal?

	class base {
	public:
		overload member;
		void member ();
		void member (int i);
	};

		Yes, but unnecessary even in older C++ implementations:
		member functions are always (potentially) overloaded.

Is the following legal?

	int global_item;
	overload global_item;
	void global_item (int i);

		No -- you can only overload functions.

Is there any possible ordering of the three statements above which would be
considered legal?

	No.

Is it ever legal to specify a return type for either a constructor or
a destructor?  Specifically, is it legal for constructors and/or
destructors to return (a) void, or (b) void*, or (c) class_type*,
where "class_type" is the name of the containing class?

	No.

Is it legal or illegal to place a declaration for a member function
within one "visibility section" (i.e. "public", "private", or "protected")
of a class declaration and then to subsequently re-declare the same
member function (with the same parameter list) in a different "visibility
section" of the same class declaration, for example:

	class base {
	protected:
		void member (int i);
	public:
		void member (int i);		// legal ?
	private:
		void member (int i) {}		// legal ??
	};

	No.  A member may be declared no more than once in a
	class declaration.  It doesn't make sense to have more
	than one visibility for a single member, anyway.

	You can, however, have several overloaded member functions
	with the same name and different visibilities:

	class Foo {
	private:
		void f(int);
	public:
		void f();
	};

	Here, Foo::f() and Foo::f(int) are two different members and
	can have different visibilities.

	A fundamental principle is that changing visibility cannot
	affect the meaning of a program -- it can only affect
	whether or not the program is considered valid.
-- 
				--Andrew Koenig
				  ark@europa.att.com

rfg@riunite.ACA.MCC.COM (Ron Guilmette) (05/03/89)

I would like to extend my warm thanks to Andrew Koenig for taking the time
to formulate concise (and precise) answers to my second set of nit picking
questions.  I hope that other readers of this newsgroup will benefit from
the public availablity of this information at least as much as I have.  I
also hope that Bjarne may be able to make some use of my tendency to nit-pick.
Specifically, I hope that some of the questions and answers exchanged here
will help to make the forthcomming new reference manual a more robust
source of information for other nit-pickers like me.

Although Andrew's answers were all quite clear and to the point, there are
one or two minor nits I feel compelled to re-pick.  The re-pickings follow
below:

In article <9279@alice.UUCP> ark@alice.UUCP (Andrew Koenig) writes:
>Here's my try at answering Ron's questions.  My comments are indented:
>------------------------------------------------------------------------
>
>	Summer.  We're still on target for June 30.

Why not just round it off and call it Independence Day?  That's pretty
close but gives you about an extra week? :-)

>How can you declare a function which can accept a pointer
>to itself as an argument (in a type-safe manner of course)?
>
>	You can't in C++ or C.

Well, so is anybody (other than me) concerned about this?

>Is it legal to declare an operator which takes no
>struct/class parameters but which does take at least
>one union parameter?  At least one enum parameter?
>
>	A union is a class, so yes. An enumeration isn't a class, so no.

OK. So is there any reason (good or otherwise) for the rule that
says that that each user-defined operator must accept at least one
struct/class/union parameter?  Is there any reason why this rule
could not be relaxed so as to say that each operator must accept at
least one struct/class/union/enum?

>If a constructor makes calls to virtual member functions
>which are declared (and defined) for the constructor's
>own class, do these calls ever go through the virtual
>function table?  If so why?
>
>	Suppose that class D is derived from B.
>	Then constructing a D involves constructing a B first.
>
>	Objects are constructed in exactly the same way
>	regardless of whether they are part of other objects.
>	Thus while the B is being constructed, it has no
>	knowledge of whether or not it's part of a D.

Is this an exception to a more general rule?  What I mean is this:
Is it also true to say that "Objects are *operated on* in exactly
the same way regardless of whether they are part of other objects."

Please correct me if I am wrong (I often am when it comes to C++) but
if a base class contains two virtual member functions, and if the second
one contains a call to the first one, and if the second one is invoked
for an object of a derived class which has the first virtual function
overloaded, which version of the first (virtual) function gets called?
An example:

	struct base {
		virtual first () {}
		virtual second () { second (); }
	};

	struct derived : public base {
		virtual first () {}
	};

	void test ()
	{
		derived d;

		d.second ();
	}

In this case, which version of first() will get called?  (I think I know the
answer, but I just want to be sure that things *do* work differently within
constructors, i.e. that "virtual-ness" doesn't count as much as it does normally.

>	It follows from this that if B's constructor calls
>	one of B's virtual functions, directly or indirectly,
>	the behavior is as if the B were being constructed
>	in isolation. You get the function from B or
>	one of its base classes, but never the function
>	from D or any other derived class of B.

So the call (within the constructor) to "one of B's virtual functions" obviously
*does not* go through the vtable.  Fine.  To that extent, you are obviously
correct in saying that things behave "as if B were being constructed in
isolation".  What happen however if we are constructing a "D_object" and
we thus (as a consequence) end up executing B's constructor, and then B's
constructor invokes one of B's virtual methods, and ***then*** the given
virtual method of B itself invokes another one of B's virtual methods (which
happens to also be overloaded in D)?

*** This is *not* an idle question.  You said that things behave "as if B were
being constructed in isolation" (i.e. without an knowledge of the fact that the
B being constructed is a part of a D).  Is that true ALL THE WAY, or only as
long as you are doing operations from directly within B's constructor?

>Is there any way to specify (explicitly) the order in which
>members and base classes are destructed?
>
>	They are destroyed in inverse order from construction.
>	There is no way to control the order of destruction
>	except by ordering your declarations to control the
>	order of construction.  Why would you want to?

Sorry.  This was a dumb question, but I feel a bit vindicated because I have
just noticed that your answer is somewhat incorrect. On pages 8-10 of the 1987
Workshop Proceedings the ability to specify various constructors which can
themselves specify various construction orderings is described.  However, on
page 10 (paragraph #2) when discussing order of destruction, its says: "There
is no way for the programmer to control this order".

This seems very asymmetric to me.  Is there any (good) reason for this asymmetry?

>Is it legal to declare a function as inline and later give
>either another declaration or a definition for the function
>whose heading does *not* include the keyword "inline"?
>
>	Yes.

OK. New question.  If you can do this, then following the later "non-inline"
declaration/definition should the compiler still consider the function to be
a candidate for inlining?

>Is it legal to declare a function as non-inline and later
>to redeclare it (or give a definition for it) which is inline?
>
>	Yes, but if you want to declare a function inline,
>	you must do so before the first place you use it.

Otherwise what?  Are you saying that the following code contains an error?  If
so what line is it on?

	void function ();

	void test ()
	{
		function ();
	}

	inline void function ()
	{
	}

>Are enum type values always treated like int type values
>or can they be used to disambiguate a function call, i.e.:
>
>	Enumerated types are separate types, so they can
>	be used to disambiguate a function call...

OK.  So enum type are *not* treated like int's!  Ah, ha!  Which of the
following things are legal, which are illegal?

	enum color { red, orange, yellow };

	int array[color];	// legal ?

	color c, c2;
	int i;
	char* cp;

		...
		i = c;
		c = i;
		c = -c;		// unary negation ?
		c = c + c;
		c = c - c;
		c = c * c;
		c = c | c;	// bitwise operators ?
		i = c == c2;	// simple comparisons ?
		i = c > c2;	// ordered comparisons ? 
		i = c == i;
		cp = &cp[c];	// index
		cp = cp+c;	// pointer arithmetic

>At the end of section 8.9 in the reference manual there is a short
>note about how to get the address of a particular instance of an
>overloaded function...

>	Sorry.  The only places you can do overloaded selection is
>	in assignment and initialization...

Don't be sorry.  I'm sorry for you! :-)  GNU C++ allows disambiguation
of a function reference via casting, i.e.:

	void* p = (function_of_one_int) &my_overloaded_function;

This appears to be a reasonable thing to allow, and I'm glad that GNU C++
allows it (even if cfront 2.0 may not).

>May an expression supplied as a default function argument contain a reference to:
>
>	d) the address of the function for which the formal parameter
>	   list is being specified?
>
>		If this means what I think it does, the answer is no --
>		how would you write the type of the argument?

This relates to an earlier question I had asked.  You can't do it in
a strictly type-safe way.  But how about this:

	class C;

	typedef void (C::*Cmp) (int, ...);

	class C {
	public:
		void member (int i = 99, Cmp ptr = &member) {}
	};


This brings up another question which I forgot to ask.  Is it legal to use a call
to a member function as part of the default value expression for a parameter of
another function within the same class, i.e.:

	class C {
	public:
		int member_1 () { return 99; }
		void member_2 (int i = member_1 ()) {}
	};


>Is it legal or illegal to put an "overload" statement inside a function?
>Inside a class?  A struct?  A union?  Is it ever of any value to nest
>overload statements within these contexts?
>
>	The `overload' declaration is obsolescent in 2.0 --
>	all functions are (potentially) overloaded.

This *is* surprizing.  Tell me something.  I got the impression that one reason
(in cfront 1.2, g++, etc.) that you would *avoid* using an "overload" declaration
for a given global function was that you could then be sure that the name of that
function would *not* be mangled on its way to assembly (or C) code.  This was/is
useful in cases where you want to link your C++ programs to code written in some
other language.  Are you saying that in 2.0, *all* the names of all global
functions will be mangled and that there will be no way to prevent this?

>Is it ever legal to specify a return type for either a constructor or
>a destructor?  Specifically, is it legal for constructors and/or
>destructors to return (a) void, or (b) void*, or (c) class_type*,
>where "class_type" is the name of the containing class?
>
>	No.

G++ allows you to explicitly declare the return type of constructors as type
"pointer-to-class" where "class" is the class being constructed.  Similarly,
it allows you to explicitly declare the type of any destructor to be "void".
Are you saing that G++ is wrong in both cases?

>Is it legal or illegal to place a declaration for a member function
>within one "visibility section" (i.e. "public", "private", or "protected")
>of a class declaration and then to subsequently re-declare the same
>member function (with the same parameter list) in a different "visibility
>section" of the same class declaration, for example:
>
>	class base {
>	protected:
>		void member (int i);
>	public:
>		void member (int i);		// legal ?
>	private:
>		void member (int i) {}		// legal ??
>	};
>
>	No.  A member may be declared no more than once in a
>	class declaration...

I don't believe that that is true.  Can you quote a page number for me on this?
Perhaps Bjarne will leap in at this point.  Actually, it has been my understanding
that the following is legal:

	class C {
	public:
		void member_1 (int i, int j);

		void member_2 () { member_1 (88, 99); }

		void member_1 (int i, int j = 88);

		void member_3 () { member_1 (77); }

		void member_1 (int i = 22, int j = 33)
		{
		}
	};

The Book says that you can re-declare a function and add additional default parameter
values.

Now if this *is* legal, then my original question (regarding re-declaration of member
functions in different visibility sections of a class declaration)  still needs to
be given a proper answer.

-- 
// Ron Guilmette  -  MCC  -  Experimental Systems Kit Project
// 3500 West Balcones Center Drive,  Austin, TX  78759  -  (512)338-3740
// ARPA: rfg@mcc.com
// UUCP: {rutgers,uunet,gatech,ames,pyramid}!cs.utexas.edu!pp!rfg

bvs@light.uucp (Bakul Shah) (05/03/89)

In article <193@riunite.ACA.MCC.COM> rfg@riunite.UUCP (Ron Guilmette) writes:
>>How can you declare a function which can accept a pointer
>>to itself as an argument (in a type-safe manner of course)?
>>
>>	You can't in C++ or C.
>
>Well, so is anybody (other than me) concerned about this?

A small extension to C can fix this: allow *forward* declaration
of a type name.  A later full declaration will complete the
definition.

    typedef foo;             /* forward declaration of type foo */
    typedef int (*foo)(foo); /* it is a type `function (foo) returning int' */

    int
    fun(foo f)
    {
	    ...
    }

    main()
    {
	    printf("%d\n", fun(fun));
    }

Extending an existing compiler to recognize this may not be easy
though.

-- Bakul Shah <..!{ucbvax,sun,uunet}!amdcad!light!bvs>

ark@alice.UUCP (Andrew Koenig) (05/03/89)

In article <193@riunite.ACA.MCC.COM>, rfg@riunite.ACA.MCC.COM (Ron Guilmette) writes:
> I would like to extend my warm thanks to Andrew Koenig for taking the time
> to formulate concise (and precise) answers to my second set of nit picking
> questions.  I hope that other readers of this newsgroup will benefit from
> the public availablity of this information at least as much as I have.

Thank you.

> I
> also hope that Bjarne may be able to make some use of my tendency to nit-pick.

Absolutely.  He reads this stuff too.

> Specifically, I hope that some of the questions and answers exchanged here
> will help to make the forthcomming new reference manual a more robust
> source of information for other nit-pickers like me.

I do too.

> Although Andrew's answers were all quite clear and to the point, there are
> one or two minor nits I feel compelled to re-pick.

No problem.

> Why not just round it off and call it Independence Day?  That's pretty
> close but gives you about an extra week? :-)

Actually we have to have final code ready well before then
so the people who duplicate tapes, print manuals, etc.
can do their thing.

> >How can you declare a function which can accept a pointer
> >to itself as an argument (in a type-safe manner of course)?
> >
> >	You can't in C++ or C.
> 
> Well, so is anybody (other than me) concerned about this?

Sure.

But suppose C++ came up with a syntax for doing such a thing.
How would you translate it into C if C can't express it?

In such knotty cases, C++ tends to take the conservative
course -- don't allow whatever it is until there's a strong
argument for doing so.  I haven't seen many applications for
passing functions to themselves, perhaps in part because you
can't do so in C in a type-safe way.

Of course, there are various cheats with ... you can use if
you really need to do it.

> OK. So is there any reason (good or otherwise) for the rule that
> says that that each user-defined operator must accept at least one
> struct/class/union parameter?

Yes.  The motivation is to make it impossible to redefine
the meaning of built-in operators, types, etc., as an
aid toward preserving programmer sanity.

> could not be relaxed so as to say that each operator must accept at
> least one struct/class/union/enum?

There's a very casual built-in conversion from enum to int, which
implicitly defines a number of built-in operators with enum arguments:

	enum e { a, b, c, d };

	e x = b;

	x++;

If it were possible to redefine these operators on enum arguments,
that would be changing the meaning of built-in things.  C++ shies
away from that.

> An example:

> 	struct base {
> 		virtual first () {}
> 		virtual second () { second (); }
> 	};

> 	struct derived : public base {
> 		virtual first () {}
> 	};

> 	void test ()
> 	{
> 		derived d;

> 		d.second ();
> 	}

> In this case, which version of first() will get called?

I presume that you intended base::second() to call first() and
not second() which would be a recursion loop.

If so, it calls derived::first().

> So the call (within the constructor) to "one of B's virtual functions" obviously
> *does not* go through the vtable.  Fine.  To that extent, you are obviously
> correct in saying that things behave "as if B were being constructed in
> isolation".  What happen however if we are constructing a "D_object" and
> we thus (as a consequence) end up executing B's constructor, and then B's
> constructor invokes one of B's virtual methods, and ***then*** the given
> virtual method of B itself invokes another one of B's virtual methods (which
> happens to also be overloaded in D)?

While the B that makes up the D is being constructed, it's a B.
It can call as many virtual functions as it likes, but it only
gets virtual functions from B or its base classes, never from
classes derived from B.

One way to look at it is that while B's constructor is executing,
the vtable is set to identify the object as a B.  Only after the
constructor returns is the vtable set to say it's a D.

> *** This is *not* an idle question.  You said that things behave "as if B were
> being constructed in isolation" (i.e. without an knowledge of the fact that the
> B being constructed is a part of a D).  Is that true ALL THE WAY, or only as
> long as you are doing operations from directly within B's constructor?

All the way.

> Sorry.  This was a dumb question, but I feel a bit vindicated because I have
> just noticed that your answer is somewhat incorrect. On pages 8-10 of the 1987
> Workshop Proceedings the ability to specify various constructors which can
> themselves specify various construction orderings is described.

Times change.

In C++ 2.0, the order in which sub-objects are constructed
is entirely controlled by the declaration order.

In particular:

	struct A { /* stuff */ };
	struct B { /* more stuff */ };

	struct C {
		A a;
		B b;

		C(/* args */): a(1), b(0) { /* stuff */ }
		C(/* different args */): b(1), a(0) { /* other stuff */ }

		// and so on
	};

In C++ 2.0, constructing a C object always constructs the `a'
and `b' members first, in that sequence, because `a' was declared
before `b'.

> page 10 (paragraph #2) when discussing order of destruction, its says: "There
> is no way for the programmer to control this order".

> This seems very asymmetric to me.  Is there any (good) reason for this asymmetry?

The reason to fix the construction and destruction order
is to ensure that things are destroyed in FIFO order
without having to remember, dynamically, the order in
which the sub-objects of an object were constructed.

There are many reasons why this is useful, most of them arcane.

> >Is it legal to declare a function as inline and later give
> >either another declaration or a definition for the function
> >whose heading does *not* include the keyword "inline"?

> OK. New question.  If you can do this, then following the later "non-inline"
> declaration/definition should the compiler still consider the function to be
> a candidate for inlining?

Yes.

Once you've said `inline' the compiler is allowed to try to inline it.

> >	Yes, but if you want to declare a function inline,
> >	you must do so before the first place you use it.

> Otherwise what?  Are you saying that the following code contains an error?  If
> so what line is it on?

> 	void function ();

> 	void test ()
> 	{
> 		function ();
> 	}

> 	inline void function ()
> 	{
> 	}

It is indeed an error.  The diagnostic from cfront is
`function called before defined as inline' and it points
to the call:

		function ();
	
I wouldn't be particularly upset at a C++ implementation
that put the error message on the definition of function().

> OK.  So enum type are *not* treated like int's!  Ah, ha!  Which of the
> following things are legal, which are illegal?

> 	enum color { red, orange, yellow };

> 	int array[color];	// legal ?

No -- color is a type, not a value.

> 	color c, c2;
> 	int i;
> 	char* cp;

In these examples, note that you can freely convert an
enum to int.

> 		i = c;

Yes.

> 		c = i;

No -- converting int to enum requires a cast:

		c = (color) i;

> 		c = -c;		// unary negation ?

No, because -c is an int that requires a cast to convert it
back to color.

> 		c = c + c;
> 		c = c - c;
> 		c = c * c;
> 		c = c | c;	// bitwise operators ?

These are all errors because you can't freely convert
int to color.

> 		i = c == c2;	// simple comparisons ?
> 		i = c > c2;	// ordered comparisons ? 
> 		i = c == i;
> 		cp = &cp[c];	// index
> 		cp = cp+c;	// pointer arithmetic

These are all legal.
 
> Don't be sorry.  I'm sorry for you! :-)  GNU C++ allows disambiguation
> of a function reference via casting, i.e.:

> 	void* p = (function_of_one_int) &my_overloaded_function;

> This appears to be a reasonable thing to allow, and I'm glad that GNU C++
> allows it (even if cfront 2.0 may not).

GNU G++ has many extensions and many other restrictions.

Anyway, casts of this sort are less obvious than they appear.
What about this one?

	void* p = (void*) &my_overloaded_function;

> This relates to an earlier question I had asked.  You can't do it in
> a strictly type-safe way.  But how about this:

> 	class C;

> 	typedef void (C::*Cmp) (int, ...);

> 	class C {
> 	public:
> 		void member (int i = 99, Cmp ptr = &member) {}
> 	};

No, because &member means &this->member, which is illegal.
Even if it were legal, it would be the wrong type -- you
need to say &C::member.  Even if that would work, you can't
take the address of C::member until you've defined it.

The following convoluted thing appears to work, though.
I don't recommend it, but here it is for the record:

	class C;

	typedef void (C::*Cmp) (int, ...);

	class C {
	public:
		void member (int, Cmp);
	};

	void C::member (int i = 99, Cmp p = (Cmp)&C::member) { }

Why are you trying to do this, anyway?  Are you writing
a threaded-code interpreter or something?

> 
> This brings up another question which I forgot to ask.  Is it legal to use a call
> to a member function as part of the default value expression for a parameter of
> another function within the same class, i.e.:
>
> 	class C {
> 	public:
> 		int member_1 () { return 99; }
> 		void member_2 (int i = member_1 ()) {}
> 	};

No -- this is an implicit reference to `this'
 
> This *is* surprizing.  Tell me something.  I got the impression that one reason
> (in cfront 1.2, g++, etc.) that you would *avoid* using an "overload" declaration
> for a given global function was that you could then be sure that the name of that
> function would *not* be mangled on its way to assembly (or C) code.  This was/is
> useful in cases where you want to link your C++ programs to code written in some
> other language.  Are you saying that in 2.0, *all* the names of all global
> functions will be mangled and that there will be no way to prevent this?

No -- in 2.0 you say which functions are C rather than saying
implicitly which ones aren't.  For example:

	extern "C" double sqrt(double);

That says that sqrt(double) is a C function and should be
compiled in whatever way is acceptable to your local C implementation.

You can still overload it without formality:

	extern "C" double sqrt(double);
	extern complex sqrt(complex);

Of course you can only say extern "C" for a single function of
a given name.  Particular implementations can assign whatever
meaning they like to the quoted string, except that every C++
implementation is expected to allow "C++" and every one that
intends to coexist with C must allow "C".  One can imagine
"Pascal", "Fortran", and so on.

> G++ allows you to explicitly declare the return type of constructors as type
> "pointer-to-class" where "class" is the class being constructed.  Similarly,
> it allows you to explicitly declare the type of any destructor to be "void".
> Are you saing that G++ is wrong in both cases?

It's either wrong or an extension, depending on your point of view.

> I don't believe that that is true.  Can you quote a page number for me on this?
> Perhaps Bjarne will leap in at this point.  Actually, it has been my understanding
> that the following is legal:
> 
> 	class C {
> 	public:
> 		void member_1 (int i, int j);
> 
> 		void member_2 () { member_1 (88, 99); }
> 
> 		void member_1 (int i, int j = 88);
> 
> 		void member_3 () { member_1 (77); }
> 
> 		void member_1 (int i = 22, int j = 33)
> 		{
> 		}
> 	};

I'll ask him.

I never write code like this, so I haven't had to think about it.
-- 
				--Andrew Koenig
				  ark@europa.att.com

bart@videovax.tv.Tek.com (Bart Massey) (05/05/89)

In article <9291@alice.UUCP> ark@alice.UUCP (Andrew Koenig) writes:
> In article <193@riunite.ACA.MCC.COM>, rfg@riunite.ACA.MCC.COM (Ron Guilmette) writes:
> > >How can you declare a function which can accept a pointer
> > >to itself as an argument (in a type-safe manner of course)?
> > >
> > >	You can't in C++ or C.
> > 
> > Well, so is anybody (other than me) concerned about this?
> 
> Sure.
> 
> But suppose C++ came up with a syntax for doing such a thing.
> How would you translate it into C if C can't express it?

Sigh.  You've already done the syntactic and semantic analysis once, in your
translator.  You don't need C's type checking, and in fact don't want it in
this case (you can always safely cast the parameter to any type you need).
So, you just make the type be (void *).  Or (int (*)()).  Or whatever.

In fact, you're probably trying to emit K&R C, for portability.  It's not
possible to declare a function which *won't* accept itself as an argument in
K&R C, since old-style function declarations don't allow argument type
checking. So...

					Bart Massey
					..tektronix!videovax.tek.com!bart
					..tektronix!reed.bitnet!bart

P.S. -- This is the kind of thing I was talking about in an earlier article
when I claimed that cfront had at least as a secondary goal the generation
of C code which corresponded closely to the original and was easily
understood by people.  As I said before, there's nothing wrong with this.
But this kind of reasoning makes it pretty apparent that this secondary goal
does exist.

jima@hplsla.HP.COM (Jim Adcock) (05/06/89)

> One can imagine "Pascal", "Fortran", and so on.

One can imagine, that since these are cstrings, implementations
might differ about the "right" "spellings" for other languages:

"C"
"C++"
"Pascal"	???
"Fortran"	???
"Algol"		???
"Modula-2"	???
"BASIC"		???
 ...
 ...
"Objective-C"	??? [juuust kidding ;-]