An enumeration is a data type consisting of a set of named values that represent integral constants, known as enumeration constants. An enumeration is also referred to as an enumerated type because you must list (enumerate) each of the values in creating a name for each of them. In addition to providing a way of defining and grouping sets of integral constants, enumerations are useful for variables that have a small number of possible values.
You can declare an enumeration type separately from the definition of variables of that type, as described in Enumeration type definition and Enumeration variable declarations; or you can define an enumeration data type and all variables that have that type in one statement, as described in Enumeration type and variable definitions in a single statement.
An enumeration type definition contains the enum, enum class, or enum struct keyword followed by an optional identifier (the enumeration tag), an optional underlying type specifier, and a brace-enclosed list of enumerators. For scoped enumerations, the identifier is mandatory. A comma separates each enumerator in the enumerator list. C99 allows a trailing comma between the last enumerator and the closing brace. XL C++ also supports this feature, for compatibility with C99.
Unscoped enumeration definition syntax >>-enum--+-+----------------+--+-------------------+--------------------+--> | '-tag_identifier-' '-:--type_specifier-' | '-nested_name_specifier--::--tag_identifier--:--type_specifier-' .-,----------. V | >--{----enumerator-+--}--;-------------------------------------><
Scoped enumeration definition syntax >>-+-enum class--+--+---------------------------+--tag_identifier--> '-enum struct-' '-nested_name_specifier--::-' .-,----------. V | >--+-------------------+--{----enumerator-+--}--;-------------->< '-:--type_specifier-'
Enumeration forward declaration syntax >>-+-enum--tag_identifier--:--type_specifier----------------+---> '-+-enum class--+--tag_identifier--+-------------------+-' '-enum struct-' '-:--type_specifier-' >--;-----------------------------------------------------------><
The keyword enum declares an unscoped enumeration type, whose enumerators are unscoped enumerators. The keywords enum class or enum struct declare a scoped enumeration type, whose enumerators are scoped enumerators. The keywords enum class and enum struct are semantically equivalent.
For the unscoped enumeration, each enumerator is declared both in the scope of the enumeration specifier, and the scope that immediately contains the enumeration specifier.
enum class Colour { orange, green, purple };
enum class Fruit { apple, pear, grape, orange };
void functionx(int) {}
void functionx(Colour) {}
void functionx(Fruit) {}
int main()
{
functionx(green); // error, green is not introduced into
// the global scope
functionx(Colour::green); // calls functionx(Colour)
functionx(Fruit::apple); // calls functionx(Fruit)
int i = Colour::orange; // error, no conversion from Colour to int
int j = (int)Colour::green; // valid, explicit cast to integer is allowed
}
The tag_identifier gives a name to the enumeration type. If you do not provide a tag name, you must put all variable definitions that refer to the enumeration type within the declaration of the type, as described in Enumeration type and variable definitions in a single statement. Similarly, you cannot use a type qualifier with an enumeration definition; type qualifiers placed in front of the enum keyword can only apply to variables that are declared within the type definition.
// the underlying type is "unsigned int"
enum Colour : unsigned int { orange = 0, green = 1, purple = 0x02U };
// the underlying type is "unsigned long"
enum class E : unsigned long { E1 = 0, E2 = 1, Emax = 0xFFFFFFF0U };
// if the type_specifier is not specified, the underlying type
// is "int" for the scoped enumeration
enum class A { A1, A2, A3 = 100, A4 };
// the underlying type is an integer type enough to represent the given
// enumerator values, eg. [signed/unsigned] long long
enum A { A1, A2, A3 = 100000000000000000000, A4 };
class C{
//Declaration of a scoped enumeration A.
enum class A;
};
// C is the nested_name_specifier that refers to the class where
// the enumeration A was declared.
enum class C::A { a, b };
You can forward declare an enumeration without providing a list of enumerators. The forward declaration of an enumeration can reduce compile time and dependencies, because it can physically decouple the implementation specifics of the enumeration and its usage.
enum E1 : unsigned long; // valid, forward declaration of an unscoped
// enumeration with the underlying type
// "unsigned long"
enum class E2 : char; // valid, forward declaration of a scoped
// enumeration with the underlying type
// "char"
enum class E3; // valid, forward declaration of a scoped
// enumeration with the implied underlying
// type "int"
enum E4; // error, you must specify the underlying type
// when you forward declare an unscoped
// enumeration
enum E1 : unsigned long; // valid, the redeclaration of E1 matches the
// previous declaration of E1
enum class E2 : short; // error, the previously declared enumeration
// E2 had the underlying type "char", and it
// cannot be redeclared with a different
// underlying type
enum E3 : int; // error, the previously declared enumeration
// E3 was a scoped enumeration, and it cannot
// be redeclared as an unscoped enumeration
enum class E3 : int { a, b, c };// valid, definition of E3
The elaborated type specifier refers to a previously declared enumeration. The x is a variable that has the type tag_identifier.
// a scoped enumeration
enum class color { red, white, black, yellow };
// an unscoped enumeration
enum letter {A, B, C, D};
// valid, regular type name usage
color pic1 = color :: white;
// valid, elaborated type usage
enum color pic2 = color :: red;
enum class color pic3 = color :: black; // invalid
enum letter let1 = letter :: A; // valid
The list of enumeration members, or enumerators, provides the data type with a set of values.
Enumeration member declaration syntax >>-identifier--+-------------------------+--------------------->< '-=--enumeration_constant-'
In C, an enumeration constant is of type int. If a constant expression is used as an initializer, the value of the expression cannot exceed the range of int (that is, INT_MIN to INT_MAX as defined in the header limits.h).
In C++, each enumeration constant has a value that can be promoted to a signed or unsigned integer value and a distinct type that does not have to be integral. You can use an enumeration constant anywhere an integer constant is allowed, or anywhere a value of the enumeration type is allowed.
The value of an enumeration constant is determined in the following way:
enum grain { oats, wheat, barley, corn, rice };
/* 0 1 2 3 4 */
enum grain { oats=1, wheat, barley, corn, rice };
/* 1 2 3 4 5 */
enum grain { oats, wheat=10, barley, corn=20, rice };
/* 0 10 11 20 21 */
enum status { run, clear=5, suspend, resume, hold=6 };
/* 0 5 6 7 6 */
func()
{
enum score { poor, average, good };
enum rating { below, average, above };
int poor;
}
With scoped enumerations, you can avoid such an error, because
the enumerators are declared only in the scope of the enumeration,
not the scope containing the enumeration.You must declare the enumeration data type before you can define a variable having that type.
Enumeration variable declaration syntax .-----------------------------. V | >>---+-------------------------+-+--enum--tag_identifier--------> +-storage_class_specifier-+ '-type_qualifier----------' >--declarator--------------------------------------------------><The tag_identifier indicates the previously-defined data type of the enumeration.
The keyword enum is optional in enumeration variable declarations.
register enum score { poor=1, average, good } rating = good;
enum score { poor=1, average, good } register rating = good;
enum score { poor=1, average, good };
register enum score rating = good;
Both examples define the enumeration data type score and the variable rating. rating has the storage class specifier register, the data type enum score, and the initial value good.
enum { Sunday, Monday, Tuesday, Wednesday, Thursday, Friday,
Saturday } weekday;
defines the variable weekday, which can be assigned any of the specified enumeration constants. However, you cannot declare any additional enumeration variables using this set of enumeration constants.