pragma once

У мовах програмування С та C++, #pragma once — нестандартна, однак широкопідтримувана директива препроцесора, створена для того, аби забезпечити включення тіла файлу сирцевого коду (зазвичай заголовного файлу) лише один раз в одній одиниці трансляції. Таким чином, #pragma once виконує ту ж функцію, що і #include guard, але з деякими перевагами, зокрема меншою кількістю коду, уникненням колізії імен та прискоренням швидкості компіляції (в окремих випадках).[1]

Приклад

Файл «grandparent.h»
#pragma once

struct foo 
{
    int member;
};
Файл «parent.h»
#include "grandparent.h"
Файл «child.c»
#include "grandparent.h" // Буде оброблено визначення структури foo
#include "parent.h"      // Файл grandparent.h вже не включатиметься

Переваги

Найпоширенішою альтернативою #pragma once є використання директиви #define для створення include guard-макроса, ім'я якого програміст вибирає унікальним для кожного файлу. Наприклад,

#ifndef GRANDPARENT_H
#define GRANDPARENT_H
// ... вміст grandparent.h
#endif /* !GRANDPARENT_H */

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

Використання #pragma once замість include guards може, для деяких компіляторів, покращити швидкість компіляції, бо це механізм вищого рівня. Компілятор може порівняти імена файлів або їхні inode без необхідності сканування препроцесором C заголовочних файлів у пошуку директив #ifndef та #endif. Важливо відзначити, що такі компілятори, як GCC, Clang та компілятори на основі EDG мають специфічні процедури для розпізнавання та оптимізації обробки include guards, тому використання #pragma once дає невелике прискорення або ж зовсім не пришвидшує компіляцію.[2][3][4]

Недоліки

Ідентифікація того ж файлу на файловій системі може бути непростою задачею[5]. Символьні та, особливо, жорсткі посилання можуть спричинити те, що один і той же файл існує під різними іменами у різних каталогах. Компілятор може використовувати евристичні методи, які порівнюють розмір, час модифікації та вміст файлів[6]. Це може призвести до протилежного ефекту коли файл скопійовано до різних частин проекту. При виконанні директиви #pragma once ці файли можуть розглядатися як різні або як один і той же компіляторо-залежним чином.

Сумісність

Compiler #pragma once
Clangпідтримується[7]
Comeau C/C++підтримується[8]
C++ Builderпідтримується[9] (з XE3)
Digital Mars C++підтримується[10]
GCCпідтримується[11] (з 3.4[5])
HP C/aC++підтримується[12] (принаймні з A.06.12)
IBM XL C/C++підтримується[13] (з 13.1.1)
Intel C++ Compilerпідтримується[14]
Microsoft Visual C++підтримується[15] (з 4.2)
Pelles Cпідтримується[16]
ARM DS-5підтримується[17]
IAR C/C++підтримується[18]
Solaris Studio C/C++не підтримується[19][20]

Примітки

  1. Noel Llopis (25.01.2005). Games from Within: Even More Experiments with Includes. Games from Within. Архів оригіналу за 30.09.2008. Процитовано 19.08.2013.
  2. The C Preprocessor: 1. The C Preprocessor. Gcc.gnu.org. 1 лютого 1996. Процитовано 19 серпня 2013.
  3. "Clang" CFE Internals Manual — Clang 3.4 documentation. Clang.llvm.org. Процитовано 19 серпня 2013.
  4. clang: File manipulation routines. Clang.llvm.org. Процитовано 19 серпня 2013.
  5. GCC 3.4 Release Series — Changes, New Features, and Fixes. Gcc.gnu.org. Процитовано 19 серпня 2013.
  6. should_stack_file() function in GCC source code.
  7. clang: clang: Pragma.cpp Source File. Clang.llvm.org. Архів оригіналу за 4 квітня 2014. Процитовано 19 серпня 2013.
  8. Comeau C++ Pre-Release User Documentation: Pragmas. Comeaucomputing.com. Архів оригіналу за 11 грудня 2013. Процитовано 19 серпня 2013.
  9. #pragma once - RAD Studio XE3. Docwiki.embarcadero.com. 2 грудня 2010. Процитовано 19 серпня 2013.
  10. Pragmas. Digital Mars. Процитовано 19 серпня 2013.
  11. Alternatives to Wrapper #ifndef. Gcc.gnu.org. Процитовано 20 серпня 2013.
  12. HP aC++/HP ANSI C A.06.26 ReleaseNotes.
  13. Supported GCC pragmas. IBM. Процитовано 20 лютого 2015.
  14. Diagnostic 1782: #pragma once is obsolete. Use #ifndef guard instead.. Intel Developer Zones. Процитовано 4 грудня 2013.
  15. once (C/C++). Msdn.microsoft.com. Процитовано 19 серпня 2013.
  16. IDE help/documentation
  17. ARM Information Center. ARM. Процитовано 17 грудня 2013.
  18. IAR C/C++ Development Guide. IAR Systems. Архів оригіналу за 16 травня 2017. Процитовано 4 грудня 2013.
  19. Solaris Studio 12.4: C++ User's Guide. Oracle. Процитовано 20 лютого 2015.
  20. Solaris Studio 12.4: C User's Guide. Oracle. Процитовано 20 лютого 2015.
This article is issued from Wikipedia. The text is licensed under Creative Commons - Attribution - Sharealike. Additional terms may apply for the media files.