15.15. Операция запятая Выражение-с-запятой: выражение , выражение
200
Пара выражений, разделенных запятой, вычисляется слева направо и значение левого выражения отбрасывается. Типом и значением результата является тип и значение правого операнда.
Эта операция группируется слева направо. В контексте, где запятая имеет специальное значение, как, например, в списке фактических аргументов функций (п. 15.1) Или в списках инициализаторов (п. 16.6), Операция запятая, описываемая в этом разделе, может появляться только в круглых скобках; например, функция
F(A,(T=3,T+2),C) имеет три аргумента, второй из которых имеет значение 5.
16. Описания Описания используются для указания интерпретации, которую язык “C” будет давать каждому идентификатору; они не обязательно резервируют память, соответствующую идентификатору. Описания имеют форму Описание: спецификаторы-описания список-описателей необ;
Описатели в списке описателей содержат описываемые идентификаторы. Спецификаторы описания представляют собой последовательность спецификаторов типа и спецификаторов класса памяти.
Спецификаторы-описания: спецификатор-типа спецификаторы-описания необ спецификатор-класса-памяти спецификатор-описания необ
список должен быть самосогласованным в смысле, описываемом ниже.
16.1. Спецификаторы класса памяти Ниже перечисляются спецификаторы класса памяти: Спецификатор-класса-памяти: AUTO STATIC EXTERN REGISTER TYPEDEF
Спецификатор TYPEDEF не реализует памяти и называется “спецификатором класса памяти” только по синтаксическим соображениям; это обсуждается в п. 16.8. Смысл различных классов памяти был обсужден в п. 12.
Описания AUTO, STATIC и REGISTER служат также в качестве определений в том смысле, что они вызывают резервирование нужного количества памяти. В случае EXTERN должно присутствовать внешнее определение (п. 18) Указываемых идентификаторов где-то вне функции, в которой они описаны.
Описание REGISTER лучше всего представлять себе как описание AUTO вместе с намеком компилятору, что описанные таким образом переменные будут часто использоваться. Эффективны только несколько первых таких описаний. Кроме того, в регистрах могут храниться только переменные определенных типов;
на PDP-11 это INT, CHAR или указатель. Существует и другое ограничение на использование регистровых переменных: к ним нельзя применять операцию взятия адреса &. При разумном использовании регистровых описаний можно ожидать получения меньших по размеру и более быстрых программ, но улучшение в будущем генерирования кодов может сделать их ненужными.
Описание может содержать не более одного спецификатора класса памяти. Если описание не содержит спецификатора класса памяти, то считается, что он имеет значение AUTO, если описание находится внутри некоторой функции, и EXTERN в противном случае. исключение: функции никогда не бывает автоматическими.
16.2. Спецификаторы типа Ниже перечисляются спецификаторы типа.
Спецификатор-типа: CHAR SHORT INT LONG UNSIGNED FLOAT DOUBLE спецификатор-структуры-или-объединения определяющее-тип-имя
Слова LONG, SHORT и USIGNED можно рассматривать как прилагательные; допустимы следующие комбинации: SHORT INT LONG INT USIGNED INT LONG FLOAT
Последняя комбинация означает то же, что и DOUBLE. В остальном описание может содержать не более одного спецификатора типа. Если описание не содержит спецификатора типа, то считается, что он имеет значение INT.
Спецификаторы структур и объединений обсуждаются в п.
16.5; Описания с определяющими тип именами TYPEDEF обсуждаются в п. 16.8.
16.3. Описатели Входящий в описание список описателей представляет собой последовательность разделенных запятыми описателей, каждый из которых может иметь инициализатор.
Список-описателей: инициализируемый-описатель инициализируемый-описатель, список-описателей инициализируемый-описатель: описатель-инициализатор необ Инициализаторы описываются в п. 16.6. Спецификаторы и описания указывают тип и класс памяти объектов, на которые ссылаются описатели. Описатели имеют следующий синтаксис:
описатель: идентификатор ( описатель ) описатель описатель () описатель [константное-выражение необ] Группирование такое же как и в выражениях.
16.4. Смысл описателей Каждый описатель рассматривается как утверждение того, что когда конструкция той же самой формы, что и описатель, появляется в выражении, то она выдает объект указанного типа и указанного класса памяти. Каждый описатель содержит ровно один идентификатор; это именно тот идентификатор, который и описывается.
Если в качестве описателя появляется просто идентификатор, то он имеет тип, указываемый в специфицирующем заголовке описания.
Описатель в круглых скобках идентичен описателю без круглых скобок, но круглые скобки могут изменять связи в составных описателях. Примеры смотри ниже.
Представим себе описание T DI где T - спецификатор типа (подобный INT и т.д.), а DI - описатель. Предположим, что это описание приводит к тому, что соответствующий идентификатор имеет тип “...T”, где “...” пусто, если DI просто отдельный идентификатор (так что тип X в “INT X” просто INT). Тогда , если DI имеет форму
*D то содержащийся идентификатор будет иметь тип “... Указатель на T”.
Если DI имеет форму D() то содержащийся идентификатор имеет тип “... Функция, возвращающая T”.
Если DI имеет форму D[константное-выражение] или D[ ] то содержащийся идентификатор имеет тип “...массив T”. В первом случае константным выражением является выражение, значение которого можно определить во время компиляции и которое имеет тип INT. (Точное определение константного выражения дано в п. 23). Когда несколько спецификаций вида “массив из” оказываются примыкающими, то создается многомерный массив; константное выражение, задающее границы массивов, может отсутствовать только у первого члена этой последовательности. Такое опускание полезно, когда массив является внешним и его фактическое определение, которое выделяет память, приводится в другом месте. Первое константное выражение может быть опущено также тогда, когда за описателем следует инициализация. В этом случае размер определяется по числу приведенных инициализируемых элементов.
Массив может быть образован из элементов одного из основных типов, из указателей, из структур или объединений или из других массивов (чтобы образовать многомерный массив).
Не все возможности, которые разрешены с точки зрения указанного выше синтаксиса, фактически допустимы. Имеются следующие ограничения: функции не могут возвращать массивы, структуры, объединения или функции, хотя они могут возвращать указатели на такие вещи; не существует массивов функций, хотя могут быть массивы указателей на функции. Аналогично, структуры или объединения не могут содержать функцию, но они могут содержать указатель на функцию.
В качестве примера рассмотрим описание INT I, *IP, F(), *FIP(), (*PFI)();
в котором описывается целое I, указатель IP на целое, функция F, возвращающая целое, функция FIP, возвращающая указатель на целое, и указатель PFI на функцию, которая возвращает целое. Особенно полезно сравнить два последних описателя.
Связь в *FIP() можно представить в виде *(FIP()), так что описанием предполагается, а такой же конструкцией в выражении требуется обращение к функции FIP и последующее использование косвенной адресации для выдачи с помощью полученного результата (указателя) целого. В описателе (*PFI)() дополни
тельные скобки необходимы, поскольку они точно так же, как и в выражении, указывают, что косвенная адресация через указатель на функцию выдает функцию, которая затем вызывается;
эта вызванная функция возвращает целое.
В качестве другого примера приведем описание FLOAT FA[17], *AFP[17];
в котором описывается массив чисел типа FLOAT и массив указателей на числа типа FLOAT. Наконец,
STATIC INT X3D[3][5][7];
описывает статический трехмерный массив целых размером 3*5*7. более подробно, X3D является массивом из трех элементов; каждый элемент является массивом пяти массивов; каждый последний массив является массивом из семи целых. Каждое из выражений X3D, X3D[I], X3D[I][J] и X3D[I][J][K] может разумным образом появляться в выражениях. Первые три имеют тип “массив”, последнее имеет тип INT.
16.5. Описание структур и объединений Структура - это объект, состоящий из последовательности именованных членов. каждый член может быть произвольного типа. Объединение - это объект, который в данный момент может содержать любой из нескольких членов. Спецификаторы и объединения имеют одинаковую форму.
Спецификатор-структуры-или-объединения
структура-или-объединение \( список-описаний-структуры\) идентификатор структуры-или-объединения \(список-описаний-структуры\) идентификатор структуры-или-объединения
Структура-или-объединение: STRUCT UNION Список-описаний-структуры является последовательностью описаний членов структуры или объединения:
Список-описаний-структуры: описание-структуры описание-структуры список-описаний-структуры описание-структуры: спецификатор-типа список-описателей-структуры список-описателей-структуры: описатель-структуры описатель-структуры, список-описателей-структуры
В обычном случае описатель структуры является просто описателем члена структуры или объединения. Член структуры может также состоять из специфицированного числа битов. Такой член называется также полем; его длина отделяется от имени поля двоеточием.
Описатель-структуры: описатель описатель: константное выражение : константное выражение
Внутри структуры описанные в ней объекты имеют адреса, которые увеличиваются в соответствии с чтением их описаний слева направо. Каждый член структуры, который не является полем, начинается с адресной границы, соответствующей его типу;