Прагма однажды - Pragma once

в C и C ++ языки программирования, прагма однажды нестандартный, но широко поддерживаемый директива препроцессора разработан, чтобы вызвать ток исходный файл для включения только один раз в одну компиляцию.[1] Таким образом, #pragma once служит той же цели, что и включить охранников, но с рядом преимуществ, в том числе: меньший объем кода, предотвращение конфликтов имен, а иногда и повышение скорости компиляции.[2] С другой стороны, #pragma once не обязательно доступен во всех компиляторах, его реализация сложна и не всегда может быть надежной.

пример

Файл "grandparent.h"
#pragma onceструктура фу {    int член;};
Файл "parent.h"
#включают "grandparent.h"
Файл "child.c"
#включают "grandparent.h"#включают "parent.h"

В этом примере включение grandparent.h в обоих parent.h и child.c обычно вызывает ошибку компиляции, потому что структура с данным именем может быть определен только один раз в данной компиляции. В #pragma once директива служит, чтобы избежать этого, игнорируя последующие включения grandparent.h.

Преимущества

С помощью #pragma once позволяет Препроцессор C для включения файла заголовка, когда это необходимо, и игнорирования #включают директива в противном случае. Это приводит к изменению поведения Препроцессор C сам по себе, и позволяет программистам выражать зависимости файлов простым способом, устраняя необходимость ручного управления.

Самая распространенная альтернатива #pragma once использовать #определять установить #include guard макрос, имя которого выбирается программистом как уникальное для этого файла. Например,

#ifndef GRANDPARENT_H#define GRANDPARENT_H... содержание из дедушка и бабушка.час#endif / *! GRANDPARENT_H * /

Такой подход минимально гарантирует, что содержимое включаемого файла не будет отображаться более одного раза. Это более подробный вариант, требует большего ручного вмешательства и подвержен ошибкам программиста, поскольку у компилятора нет доступных механизмов для предотвращения случайного использования одного и того же имени макроса более чем в одном файле, что может привести к созданию только одного из файлов. включены. Такие ошибки вряд ли останутся незамеченными, но могут усложнить интерпретацию отчета об ошибках компилятора. Поскольку препроцессор сам отвечает за обработку #pragma once, программист не может делать ошибки, которые вызывают конфликты имен.

В отсутствие # включить охранников около #включают директивы, использование #pragma once улучшит скорость компиляции для некоторых компиляторов, так как это механизм более высокого уровня; сам компилятор может сравнивать имена файлов или inodes без обращения к Препроцессор C сканировать заголовок на предмет #ifndef и #endif. Тем не менее, поскольку защита включения появляется очень часто и накладные расходы на открытие файлов значительны, компиляторы обычно оптимизируют обработку защиты включения, делая их так же быстро, как #pragma once.[3][4][5]

Предостережения

Определить один и тот же файл в файловой системе - нетривиальная задача.[6] Символические ссылки и особенно жесткие ссылки могут привести к тому, что один и тот же файл будет найден под разными именами в разных каталогах. Компиляторы могут использовать эвристику, которая сравнивает размер файла, время модификации и содержимое.[7] Дополнительно, #pragma once может поступить неправильно, если один и тот же файл намеренно скопирован в несколько частей проекта, например при подготовке сборки. В то время как включить охранников все равно защитит от двойных определений, #pragma once может или не может рассматривать их как один и тот же файл в зависимости от компилятора. Эти трудности, вместе с трудностями, связанными с определением того, что составляет один и тот же файл при наличии жестких ссылок, сетевых файловых систем и т. Д., До сих пор препятствовали стандартизации #pragma once.[нужна цитата ]

Использование #include guard макросы позволяют зависимому коду распознавать небольшие различия в семантике или интерфейсах конкурирующих альтернатив и реагировать на них. Например,

#включают TLS_API_MACRO / * определено в командной строке * /...#if defined TLS_A_H... использовать один известен API#elif определил TLS_B_H... использовать еще один известен API#else#error "нераспознанный TLS API"#endif

В этом случае прямое определение того, какой API доступен, будет использовать тот факт, что включаемый файл рекламировал себя своим #include guard макрос.

В #включают Директива определена, чтобы представить намерение программиста фактически включить текст файла в точку директивы. Это может происходить несколько раз в пределах одной единицы компиляции и полезно для многократной оценки содержимого, содержащего макрос, на предмет изменения определений макроса.

Использование #pragma once, как использование #include guard макросы внутри включаемого файла возлагают ответственность на его авторов, чтобы защитить себя от нежелательного множественного включения. Чрезмерная зависимость программистов от любого механизма путем прямого, незащищенного использования #включают директивы без собственных #include guard приведет к сбою при использовании включаемого файла, не защищенного ни одним из механизмов.

Портативность

Компилятор#pragma once
ЛязгПоддерживается[8]
Комо C / C ++Поддерживается[9]
Cray C и C ++Поддерживается[10] (с 9.0)
C ++ Builder XE3Поддерживается[11]
Цифровой Марс C ++Поддерживается[12]
GCCПоддерживается[13] (официально с 3.4[6][14])
HP C / aC ++Поддерживается[15] (начиная с A.06.12)
IBM XL C / C ++Поддерживается[16] (начиная с 13.1.1)
Компилятор Intel C ++Поддерживается[17][неудачная проверка ]
Microsoft Visual C ++Поддерживается[18][19] (начиная с 4.2)
Компилятор NVIDIA CUDAПоддерживается (в зависимости от базового компилятора хоста)
Пеллес СПоддерживается[20]
АРМ ДС-5Поддерживается[21]
IAR C / C ++Поддерживается[22]
Keil CC 5Поддерживается[23]
Oracle Developer Studio C / C ++Поддерживается[24] (с 12.5)
Портлендская группа C / C ++Поддерживается[25] (начиная с 17.4)
TinyCCПоддерживается[26] (с апреля 2015 г.)
TASKING VX-toolset для TriCore: компилятор CПоддерживается[27] (начиная с v6.2r2)

Рекомендации

  1. ^ "однажды". Документы Microsoft. 3 ноября 2016 г.. Получено 25 июля 2019.
  2. ^ «Игры изнутри: еще больше экспериментов с включениями». Web.archive.org. 2005-01-25. Архивировано из оригинал 30 сентября 2008 г.. Получено 2013-08-19.
  3. ^ "Препроцессор C: 1. Препроцессор C". Gcc.gnu.org. 1996-02-01. Получено 2013-08-19.
  4. ^ ""Clang "Руководство по внутреннему устройству CFE - документация Clang 3.4". Clang.llvm.org. Получено 2013-08-19.
  5. ^ "clang: процедуры обработки файлов". Clang.llvm.org. Получено 2013-08-19.
  6. ^ а б «Серия выпусков GCC 3.4 - Изменения, новые функции и исправления». Gcc.gnu.org. Получено 2013-08-19.
  7. ^ "функция should_stack_file () в исходном коде GCC".
  8. ^ "clang: clang: Исходный файл Pragma.cpp". Clang.llvm.org. Архивировано из оригинал на 2014-04-04. Получено 2013-08-19.
  9. ^ "Предварительная пользовательская документация Comeau C ++: Pragmas". Comeaucomputing.com. Получено 2013-08-19.[мертвая ссылка ]
  10. ^ "Обзор версии CCE 9.0.0, введение S-5212". Cray Inc. 01.06.2019. Получено 2019-09-23.
  11. ^ "#pragma once - RAD Studio XE3". Docwiki.embarcadero.com. 2010-12-02. Получено 2013-08-19.
  12. ^ «Прагмы». Цифровой Марс. Получено 2013-08-19.
  13. ^ "Альтернативы Wrapper #ifndef". Gcc.gnu.org. Получено 2013-08-20.
  14. ^ "Ошибка GCC 11569 - нет замены #pragma once". 2003-07-18. Получено 2020-10-21.
  15. ^ «Руководство программиста HP aC ++ / HP C A.06.29; март 2016 г. (AR1603)».
  16. ^ «Поддерживаемые прагмы GCC». IBM. Получено 2015-02-20.
  17. ^ «Диагностика 1782: #pragma once устарела. Вместо этого используйте #ifndef guard». Зоны разработчиков Intel. Получено 4 декабря 2013.[мертвая ссылка ]
  18. ^ "один раз (C / C ++)". Msdn.microsoft.com. Архивировано из оригинал на 2016-08-10. Получено 2013-08-19.
  19. ^ https://msdn.microsoft.com/en-us/library/4141z1cx.aspx
  20. ^ Справка / документация по IDE
  21. ^ «Информационный центр АРМ». РУКА. Получено 2013-12-17.
  22. ^ "Руководство по разработке IAR C / C ++" (PDF). IAR Systems. Архивировано из оригинал (PDF) 16 мая 2017 г.. Получено 4 декабря 2013.
  23. ^ «Прагмы, признанные компилятором». Кейл.
  24. ^ «Oracle® Developer Studio 12.5: Руководство по совместимости с GCC». Oracle. Получено 2016-07-26.
  25. ^ "Портлендская группа". Получено 31 июля 2016.
  26. ^ «Прагма TinyCC после реализации». Получено 19 июн 2018.
  27. ^ "MA160-800 (v6.2r2) 13 марта 2018 г. стр. 92" (PDF).

внешняя ссылка