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