Re: (Forward) declarations of enums
Greg Herlihy ha scritto:
On Dec 4, 3:31 pm, AlbertoBarb...@libero.it (Alberto Ganesh Barbati)
wrote:
Hi Everybody,
in a project I am working on, we have an enumeration with several
hundred enumerators that is generated by an automatic tool. Each
enumerator represent an ID of localizable string. The localizer
component is accessible from several places on in the project, however=
,
only very few compilation units actually require access to the values =
of
the enumerators. Each time the automatic tool re-generate the
enumeration, nearly half of the project have to be re-compiled, althou=
gh
only few compilation units would actually need it. A way to isolate th=
e
enumerators while keeping the type-safety provided by the enumerated
type would be very useful.
I think that the more straightforward approach would be to limit the
number of source files that include the enum definition (or references
to it, including function prototypes). Only those source files that
need the enum definition (say, to pass as an argument in a function
call) should include the necessary header.
This is not always practically possible.
The first thing that comes in mind is to "forward" declare the
enumeration (that is to declare the name of the enumeration without
listing all enumerators), like this:
enum E; // enumerators defined elsewhere: illegal in C++03
void Func(E);
A forward-declaration of the enum E would be pointless - because
either a) a source file calls Func() (and therefore has to include the
complete definition of E anyway) or b) a source file does not call
Func() - in which case there is no reason to include either the Func()
function prototype or the forward declaration of E - in the first
place.
This premise is wrong. A value of an enumeration could be obtained and
used without any knowledge of the enumerators, for example in this case:
--- fileA.h
enum Opaque : int; // suppose it were legal
Opaque RetrieveData();
void DoSomethingWithData(Opaque);
--- fileA.cpp
#include "fileA.h"
enum Opaque : int {
Value1, Value2, /*...*/
};
Opaque RetrieveData() { /*...*/ }
void DoSomething(Opaque o) { /*...*/ }
--- fileB.cpp
#include "fileA.h" // look Ma, no enumerators!
int main()
{
DoSomethingWithData(RetrieveData());
}
There are also other use cases. A very good one has been identified by
Daniel Kr=C3=BCgler, that is:
struct Something {
enum State : unit64_t; // Would be fine
State s; // Even this should work
};
Actually this scenario is very similar to the one I am facing in my
motivating project! The "s" member in my case is private and no methods
in the public interface of "Something" get or return values of type
"State". In this scenario there is no chance that a translation unit
including the definition of Something will ever need the actual list of
enumerators (except, of course, the one implementing the member
functions of Something.)
Unfortunately, that is illegal in C++03. The rationale is that without
the complete list of enumerators we can't determine the underlying typ=
e
of E. The obvious workaround is to use an integral type instead of the
enumeration and provide a hopefully clear comment:
void Func(int); // actually a value of enum E should be passed here
The rationale has nothing to do with the underlying type of the enum -
and everything to do with the fact that source files fall into either
one of the two categories mentioned above: those that require the
complete enum definition and those that do not need the enum
definition at all. Because, unlike a class object, which can be passed
by reference or pointer (and thus may be an incomplete type) - an enum
is always referenced by value. So the practice of forwardly declaring
a class - simply has no equivalent purpose when it comes to enums.
I disagree, the underlying type is the key point. If you know that, you
can manipulate values of the enum even without any knowledge of what
those values mean or how they have been obtained, as I've shown in the
my first example. Indeed, there *is* a difference between class objects
and enum, that is a declared (but not defined) class object is an
incomplete type, while an enum will always be a complete type as soon as
you know the underlying type. For this reason, it's unfair to compare
enums with class objects, the correct comparison is between enums and
*pointers* to class objects. You can manipulate pointers to incomplete
types precisely because you know the representation of pointers even if
the pointed type is incomplete.
You are describing "strong" or "opaque" typedefs. See:
http://www.open-std.org/JTC1/SC22/WG21/docs/papers/2005/n1891.pdf
http://www.open-std.org/JTC1/SC22/WG21/docs/papers/2006/n2141.html
No, I am not referring to those, although they address a similar issue.
An enumeration is both a type *and* a set of named values, while strong
typedefs are just types: that's a very big difference.
I am not sure whether strong typedefs will make it into C++ or not.
Another good reason to avoid messing with them at this stage.
Ganesh
---
[ comp.std.c++ is moderated. To submit articles, try just posting with ]
[ your news-reader. If that fails, use mailto:std-c++@ncar.ucar.edu ]
[ --- Please see the FAQ before posting. --- ]
[ FAQ: http://www.comeaucomputing.com/csc/faq.html ]