Структура языка Си - PullRequest
       23

Структура языка Си

0 голосов
/ 30 сентября 2011

Почему это работает

printf("Hello"
"World");

Принимая во внимание

printf("Hello
""World");

нет? ANSI C объединяет смежные строки, это нормально ... но это другое дело. Это как-то связано с анализатором языка C или с чем-то еще? Спасибо

Ответы [ 3 ]

7 голосов
/ 30 сентября 2011

Строка должна заканчиваться до конца строки. Это хорошая вещь. В противном случае забытая закрывающая кавычка может помешать выполнению последующих строк кода.

Это может занять значительное время для отладки. В наши дни раскраска синтаксиса должна была дать подсказку, но в первые годы были монохромные дисплеи.

1 голос
/ 30 сентября 2011

Язык C определяется в терминах токенов , и один из токенов является строковым литералом (в стандартном варианте: s-char-sequence ). s-char-последовательности начинаются и заканчиваются неэкранированными двойными кавычками и не должны содержать неэкранированные символы новой строки.

Соответствующий стандарт (C99) цитата:

> Syntax
>   string-literal:
>     " s-char-sequence(opt) "
>     L" s-char-sequence(opt) "
>   s-char-sequence:
>     s-char
>     s-char-sequence s-char
>   s-char:
>     any member of the source character set
>           except the double-quote ", backslash \,
>           or new-line character
>     escape-sequence

Экранированные символы новой строки, однако, удаляются на ранней стадии перевода, называемой объединением строк , поэтому компилятору никогда не удается их интерпретировать. Вот соответствующая стандартная (C99) цитата:

Приоритет среди синтаксических правил перевода определяется следующими этапами.

  1. Многобайтовые символы физического исходного файла отображаются, в соответствии с реализацией, в исходный набор символов (ввод символов новой строки для индикаторов конца строки), если это необходимо. Последовательности триграфа заменяются соответствующими односимвольными внутренними представлениями.
  2. Каждый экземпляр символа обратной косой черты (\), за которым сразу следует символ новой строки, удаляется, объединяя физические исходные строки для формирования логических исходных строк. Только последняя обратная косая черта в любой физической исходной строке имеет право на участие в таком соединении. Исходный файл, который не является пустым, должен заканчиваться символом новой строки, которому не должен предшествовать символ обратной косой черты, прежде чем произойдет любое такое соединение.
  3. Исходный файл разлагается на токены предварительной обработки6) и последовательности символы пробела (включая комментарии). Исходный файл не должен заканчиваться частичный токен предварительной обработки или частичный комментарий. Каждый комментарий заменяется один пробел Символы новой строки сохраняются. Будь каждый непустой последовательность символов пробела, кроме новой строки, сохраняется или заменяется одним символом пробела в зависимости от реализации.
  4. Выполняются директивы предварительной обработки, расширяются вызовы макросов и _Pragma Унарные операторные выражения выполняются. Если последовательность символов, которая соответствует синтаксису универсального имени персонажа, созданного токеном конкатенация (6.10.3.3), поведение не определено. # Включить предварительную обработку директива заставляет именованный заголовок или исходный файл обрабатываться с фазы 1 до фазы 4, рекурсивно. Все директивы предварительной обработки затем удаляются.
  5. Каждый элемент исходного набора символов и escape-последовательность в символьных константах и ​​строковых литералах преобразуются в соответствующий элемент набора символов выполнения; если соответствующего члена нет, он преобразуется в определяемый реализацией член, отличный от нулевого (широкого) символа.7)
  6. Литеральные токены смежных строк объединяются.
  7. Пробельные символы, разделяющие токены, больше не имеют значения. каждый Токен предварительной обработки преобразуется в токен. Полученные токены синтаксически и семантически проанализирован и переведен как единица перевода.
  8. Все внешние ссылки на объекты и функции разрешены. Компоненты библиотеки связаны для удовлетворения внешних ссылок на функции и объекты, не определенные в текущем переводе. Весь такой вывод транслятора собирается в образ программы, который содержит информацию, необходимую для выполнения в среде выполнения.
1 голос
/ 30 сентября 2011

Вы не можете создать новую строку в строковом литерале. Это был выбор, сделанный моими дизайнерами C. IMO, но это хорошая особенность. Однако вы можете сделать это:

printf("Hello\
""World");

Что дает те же результаты.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...