[comp.lang.c++] Structures want anonymity, too.

kraus@uiucdcsb.UUCP (03/01/87)

Anonymous unions are nice.  For example, page 67 of the C++ book
gives the following example:

    struct entry {
        char* name;
        char type;
        union {
            char* string_value;     // used if type=='s'
            int int_value;          // used if type=='i'
        };
    };

If "x" is an "entry", then one can get to the fields in the union
with "x.string_value" and "x.int_value".  (In C, one would have to
give the union a name, which is annoying.)

But why aren't there anonymous structures, too?  For example, suppose
that for some reason I also needed a flag in case type=='i'.  I would
like to say

        union {
            char* string_value;
            struct { int int_value; int flag; };
        };

But this does not seem to be allowed.  (I believe that it was version 1.1
that I tried this on.)  So, I am forced to put "int flag" at the top level
of the structure, which is annoying.  It gets worse when lots of cases
need several fields.  The clearest solution that I have found is to have
several anonymous unions in succession.

It also seems the me that adding anonymous structures would increase the
orthogonality of the language.


Tim Kraus
Department of Computer Science
University of Illinois at Urbana-Champaign
1304 West Stringfield Avenue
Urbana, IL  61801

USENET: {pur-ee,convex,ihnp4}!uiucdcs!kraus
ARPA:   kraus@b.cs.uiuc.edu
CSNET:  kraus%uiuc@csnet-relay

roger@valid.UUCP (03/12/87)

We (Valid Logic Systems, Inc.) have modified our 1.1 C++ translator to
allow anonymous structures as well as anonymous unions.  Our primary motivation
for this enhancement was for use with our Pascal-to-C++ translator that we
are developing.  In Pascal there is a construct known as a "variant record"
that takes the following form (from Jensen & Wirth, 2nd Edition, pp. 42 & 45):

<record type> ::= record <field list> end
<field list> ::= <fixed part> | <fixed part> ; <variant part> | <variant part>
<fixed part> ::= <record section> { ; <record section>}
<record section> ::= <field identifier> {, <field identifier>} : <type>|<empty>
<variant part> ::= case <tag field> <type identifier> of <variant> {; <variant>}
<variant> ::= <case label list> : ( <field list> ) | <empty>
<case label list> ::= <case label> {, <case label>}
<case label> ::= <constant>
<tag field> ::= <identifier> : | <empty>

Thus, one can have types such as:

variant_record_example = record
	a: integer;
	case boolean of
		FALSE: (b, c, d: integer; e: real);
		TRUE: (f: array [0 .. 10] of char; g: random_type_name)
	end;

var q: variant_record_example;
begin
q.a := 1;
q.b := 2;
q.e := 3.0;
<etc.>

where all of the names of the various variant and non-variant fields are in
the same scope.  The "equivalent" type in C would look like:

typedef struct {
	int a;
	union {
		struct {
			int b, c, d; float e;
		} _pointless_name_1;
		struct {
			char f[11]; random_type_name g;
		} _pointless_name_2;
	} _pointless_name_3;
} variant_record_example;

variant_record_example q;
q.a = 1;
q._pointless_name_3._pointless_name_1.b = 2;
q._pointless_name_3._pointless_name_1.e = 3;
<etc.>

Standard C++ allows you to eliminate _pointless_name_3, but still requires
_pointless_name_1 and _pointless_name_2.  Our extension allows you to declare
this type as:

typedef struct {
	int a;
	union {
		struct {
			int b, c, d; float e;
		};
		struct {
			char f[11]; random_type_name g;
		};
	};
} variant_record_example;

Once you have anonymous structures AND unions you begin to see their more
abstract function, which is to group a collection of variables for storage
allocation purposes; a struct allocates them in series and a union allocates
them in parallel.  Naming a struct or union then becomes merely a scope issue,
allowing you to create many simultaneous, independent name spaces for
variables.
I have discussed this issue with Bjarne and he said that he is considering
adopting this change in the standard.  Others who believe it is a useful
generalization might want to drop Bjarne a line to that effect.

Roger H. Scott
Valid Logic Systems, Inc.
2820 Orchard Parkway
San Jose, CA 95134
(408) 432-9400 x2484

len@geac.UUCP (Leonard Vanek) (03/16/87)

In article <1062@valid.UUCP> roger@valid.UUCP (Roger H. Scott) writes:
>We (Valid Logic Systems, Inc.) have modified our 1.1 C++ translator to
>allow anonymous structures as well as anonymous unions.  Our primary motivation
>for this enhancement was for use with our Pascal-to-C++ translator that we
>are developing.  In Pascal there is a construct known as a "variant record"
>that takes the following form (from Jensen & Wirth, 2nd Edition, pp. 42 & 45):
>
He then goes on to explain the need for the following syntax for
emulating a pascal variant record:
>
>typedef struct {
>	int a;
>	union {
>		struct {
>			int b, c, d; float e;
>		};
>		struct {
>			char f[11]; random_type_name g;
>		};
>	};
>} variant_record_example;

This is certainly the minimal change to achieve the desired
result and may be preferred for precisely that reason. However,
it strikes me that if what is wanted is a way to get variant
records in C++ (not just for Pascal to C translators but also for
original code) why not look for a way to eliminate all of the
redundant syntax to come up with a clean ``variant structure''
mechanism? Perhaps it could be modelled on the switch statement
in much the same way that Pascal and Ada model their variant
records on their case statements?

I have not thought this idea out so it may not be all that good,
but as long as Bjarne and other readers of this newsgroup are
considering the small change maybe you should consider the larger
one as well.

---------------------------------------------------------------------
Leonard Vanek                       phone (416) 475-0525
Geac Computers International
350 Steelcase Rd. West
Markham Ontario L3R 1B3
Canada

UUCP ... {allegra,ihnp4,decvax,pyramid} !utzoo!yetti!geac!len