 |
» |
|
|
|
|  |  |
In general, a variable declaration has the following format: [storage_class_specifier] [data_type] variable_name [=initial_value];
|
where: - storage_class_specifier
is an optional keyword. - data_type
is one of the data types described in Chapter 3 “Data Types and Declarations ” - variable_name
is a legal identifier. - initial_value
is an optional initializer for the variable.
Here are a few sample variable declarations without storage
class specifiers or initial values: int age; /* integer variable "age" */ int length, width; /* abbreviated declaration of two variables*/ float ph; /* floating-point variable "ph" */ char a_letter; /* character variable "a_letter" */ int values[10]; /* array of 10 integers named values */ enum days {mon, wed, fri}; /* enumerated variable "days" */
|
 |
Typedef Declarations |  |
The C language allows you to create your own names for data
types with the typedef
keyword. Syntactically, a typedef
is similar to a variable declaration except that the declaration
is preceded by the typedef
keyword. A typedef
declaration may appear anywhere a variable declaration may appear
and obeys the same scoping rules as a normal declaration. Once declared,
a typedef name
may be used anywhere that the type is allowed (such as in a declaration,
cast operation, or sizeof
operation). You can write typedef
names in all uppercase so that they are not confused with variable
names. You may not include an initializer with a typedef. The statement: typedef long int FOUR_BYTE_INT;
|
makes the name FOUR_BYTE_INT
synonymous with long int.
The following two declarations are now identical: long int j; FOUR_BYTE_INT j;
|
Abstract Global Types |  |
Typedefs
are useful for abstracting global types that can be used throughout
a program, as shown in the following structure and array declaration: typedef struct { char month[4]; int day; int year; } BIRTHDAY; typedef char A_LINE[80]; /* A_LINE is an array of * 80 characters */
|
Improving Portability |  |
Type definitions can be used to compensate for differences
in C compilers. For example: #if SMALL_COMPUTER typedef int SHORTINT; typedef long LONGINT; #elif BIG_COMPUTER typedef short SHORTINT; typedef int LONGINT; #endif
|
This is useful when writing code to run on two computers,
a small computer where an int
is two bytes, and a large computer where an int
is four bytes. Instead of using short, long, and int, you can use
SHORTINT and
LONGINT and be
assured that SHORTINT
is two bytes and LONGINT
is four bytes regardless of the machine. Simplifying Complex
Declarations |  |
You can use typedefs to simplify
complex declarations. For example: typedef float *PTRF, ARRAYF[], FUNCF();
|
This declares three new types called PTRF
(a pointer to a float), ARRAYF
(an array of floats), and FUNCF
(a function returning a float).
These typedefs
could then be used in declarations such as the following: PTRF x[5]; /* a 5-element array of pointers to floats */ FUNCF z; /* A function returning a float */
|
Using typedefs for
Arrays |  |
The following two examples illustrate what can happen when
you mix pointers and typedefs that
represent arrays. The problem with the program on the left is that
ptr points to an array of 80 chars,
rather than a single element of a char
array. Because of scaling in pointer arithmetic, the increment operator
adds 80 bytes, not one byte, to ptr. Name Spaces |  |
All identifiers (names) in a program fall into one of four
name spaces. Names in different name spaces never interfere with
each other. That is, you can use the same name for an object in
each of the four name spaces without these names affecting one another.
The four name spaces are as follows: - Structure,
Union, and Enumeration Tags
Tag names that immediately follow these type specifiers:
struct, union,
and enum. These
types are described in “Structure and Union Specifiers ”. - Member Names
Names of members of a structure or union. - Goto Labels
Names that mark the target of a goto
statement. - Function, Variable and All Other Names
Any names that are not members of the preceding
three classes.
 |  |  |  |  | NOTE: The separate name spaces for goto
labels and for each struct, union,
or enum definition are part of
the ANSI/ISO standard, but not part of the K&R
language definition. |  |  |  |  |
The following example uses the same name, overuse,
in four different ways: int main(void) { int overuse; /* normal identifier */ struct overuse { /* tag name */ float overuse; /* member name */ char *p; } x; goto overuse; overuse: overuse = 3; /* label name */ }
|
Structure, Union, and Enum NamesEach struct,
union, or enum
defines its own name space, so that different declarations can have
the same member names without conflict. The following is legal: struct A { int x; float y; }; struct B { int x; float y; };
|
The members in struct A are
distinct from the members in struct
B. Macro names do interfere with the other
four name spaces. Therefore, when you specify a macro name, do not
use this name in one of the other four name spaces. For example,
the following program fragment is incorrect because it contains
a macro named square and a label
named square: #define square(arg) arg * arg int main(void) { ... square: ... }
|
|