[comp.lang.c++] 1 Definition Allowed

kearns@cs.columbia.edu (Steve Kearns) (11/02/90)

E&S (3.1) says only one definition is allowed per program for any
name.  It also claims that "struct S { int a; int b; };" is a
definition of a struct.  But this seems to imply that such a struct
declaration cannot appear in a header file which will be included in
more than 1 compilation unit!!!!!  This is obviously counter to
anyone's intuition, so what gives?

-steve

roger@procase.UUCP (Roger H. Scott) (11/04/90)

In article <1990Nov1.202530.15347@cs.columbia.edu> kearns@cs.columbia.edu (Steve Kearns) writes:
>E&S (3.1) says only one definition is allowed per program for any
>name.  It also claims that "struct S { int a; int b; };" is a
>definition of a struct.  But this seems to imply that such a struct
>declaration cannot appear in a header file which will be included in
>more than 1 compilation unit!!!!!

Section 3 says that class/struct/union/enum names may or may not be "linked"
between files, depending on whether or not they are in any way used to 
declare objects that have external linkage.  This is pretty weird, and it
does seem to imply that, for instance, a struct used to declare a global
variable *cannot* be defined in more than one file.  Clearly this is absurd.
As I see it the problem here is that "definition" is the wrong word to use
when talking about types (such as structs).  ANSI C has the right idea here -
definitions allocate storage.  If the issues of storage allocation don't
apply to the entity in question then the word "definition" doesn't, either.
Some other term needs to be coined to describe the difference between
"struct S", which is a declaration (though not of an object), and "struct S
{ int m; }", which is "something else".  If we call this "something else" a
"complete description", then the rule would be not that any given type must
have exactly one complete description within a program, but rather that all
complete descriptions of a given type within a program must be identical.
Now it becomes a link-level semantics checking problem for an implementation
to actually enforce this rule.

horstman@sjsumcs.sjsu.edu (Cay Horstmann) (11/05/90)

In article <1990Nov1.202530.15347@cs.columbia.edu> kearns@cs.columbia.edu (Steve Kearns) writes:
>E&S (3.1) says only one definition is allowed per program for any
>name.  It also claims that "struct S { int a; int b; };" is a
>definition of a struct.  But this seems to imply that such a struct
>declaration cannot appear in a header file which will be included in
>more than 1 compilation unit!!!!!  This is obviously counter to
>anyone's intuition, so what gives?
>
This is NOT what they mean. Yes, you can (and must) have the same struct
definition in many different compilation units (i.e. files.) 

What you cannot have (and that is STUPID) is multiple definitions of the
same struct in the same file. While it is perfectly legal to have fifteen
declarations of extern int x; or int foo(int);, and the compiler will
happily check for consistency, it is not legal to have two occurrences
of struct Complex { double re,im; }; in the same file. 

I wish this were remedied. One of the best pieces of advice I know is 
TO INCLUDE EACH HEADER FILE IN ITS OWN .C FILE, but it just doesn't work
for structs, enums, and typedefs. It does not appear too difficult to check.
I agree that checking consistency of inline functions is probably 
unreasonable, and a redefinition error is reasonable there. 

Cay

pena@fuug.fi (Olli-Matti Penttinen) (11/05/90)

In article <1990Nov5.051429.6109@sjsumcs.sjsu.edu> horstman@sjsumcs.sjsu.edu (Cay Horstmann) writes:

  >What you cannot have (and that is STUPID) is multiple definitions of the
  >same struct in the same file. While it is perfectly legal to have fifteen
  >declarations of extern int x; or int foo(int);, and the compiler will
  >happily check for consistency, it is not legal to have two occurrences
  >of struct Complex { double re,im; }; in the same file. 

Quite the contrary.  It would be a colossal waste of time to check the
consistency of every type definition.  Since the line must be drawn
somewhere, I agree with ARM (and current C++ implementations) that the
best place is right where it lies now: there has to be exactly one
type definition per a user defined type per compilation unit.

  >I wish this were remedied. One of the best pieces of advice I know is 
  >TO INCLUDE EACH HEADER FILE IN ITS OWN .C FILE, but it just doesn't work
  >for structs, enums, and typedefs. It does not appear too difficult to check.
  >I agree that checking consistency of inline functions is probably 
  >unreasonable, and a redefinition error is reasonable there. 

By borrowing a bit of Modula-2 type of thinking, this problem can be
elegantly alleviated: I suggest using an additional abstraction level,
the module.  Each module consists of a (small) number of types that
are somehow logically connected (e.g. a collection and an accompanying
iterator).  The definitions of each type are placed in a single .h
file, the interface to the module.  A necessary evil is the need to
put all public/protected inline definitions in this file, as well.

Inline definitions should be apart from the class declaration body,
however.  For one thing, it's easier to find out what a class does, if
it's declaration is uncluttered, on the other hand, potential forward
reference problems can be handled this way.

In a complex situation, another .h file might be handy: any constants,
etc. that only affect the implementation and not the use of the module
as well as bodies of private inline functions should be placed in it.

Finally, definitions of member, static and global functions that
implement the class, should be scattered across 1..n .C files,
depending on their size and number.  These would naturally include (in
this order): system headers, interfaces of all modules used, the
module's own interface and the private supplemental header file.  Once
compiled, the files could be combined to a library and shipped with
the interface .h file.

One problem remains: it would still be possible to accidentally
include an interface more than once.  This can be avoided if every
interface "knows" whether it already is included or not:

    //interface.h

    #ifndef INTERFACE_H

    #define INTERFACE_H

    #include <system.h>
    #include "anothermodule.h"

    class thisclass {
        .
        .
        .
    };

    inline T thisclass::f() { ...};

    #endif

    // EOF
--
Olli-Matti Penttinen <pena@fuug.innopoli.fi>
Brainware Oy                      "When in doubt, use brute force."
Tekniikantie 17                      --Ken Thompson
02150  ESPOO, Finland   Tel. +358 0 4375 320   Fax. +358 0 4553 117