Какие типы заголовочных файлов не должны быть защищены от многократного включения?
Я прочитал исходный код dcmtk и нашел комментарий в ofstdinc.h:
// this file is not and should not be protected against multiple inclusion
И какие типы заголовочных файлов НЕ ДОЛЖНЫ быть защищены от многократного включения?
3 ответа
Препроцессорное метапрограммирование. То есть использование включенного файла в качестве некой функции времени компиляции, которая выполняет какую-то задачу. Аргументами функции являются макросы. Например, у файла, который вы связали, есть раздел, который выглядит так:
// define INCLUDE_STACK to include "ofstack.h"
#ifdef INCLUDE_STACK
#include "dcmtk/ofstd/ofstack.h"
#endif
Так что, если бы я хотел включить "ofstack.h"
Я бы сделал так:
#define INCLUDE_STACK
#include "ofstdinc.h"
#undef INCLUDE_STACK
Теперь, представьте позже, кто-то хочет использовать этот конкретный раздел заголовка:
// define INCLUDE_STRING to include "ofstring.h"
#ifdef INCLUDE_STRING
#include "dcmtk/ofstd/ofstring.h"
#endif
Поэтому они делают следующее:
#define INCLUDE_STRING
#include "ofstdinc.h"
#undef INCLUDE_STRING
Если "ofstdinc.h"
были включены охранники, это не будет включено.
Одним из примеров являются заголовочные файлы, которые ожидают, что вы определите макрос. Рассмотрим заголовок m.h
с
M( foo, "foo" )
M( bar, "bar" )
M( baz, "baz" )
Это может быть использовано в другом заголовке, например:
#ifndef OTHER_H
#define OTHER_H
namespace other
{
enum class my_enum
{
#define M( k, v ) k,
#include "m.h"
#undef M
};
void register_my_enum();
}
#endif
и в каком-то другом файле (возможно, реализация):
#include "other.h"
namespace other
{
template< typename E >
void register_enum_string( E e, const char* s ) { ... }
void register_my_enum()
{
#define M( k, v ) register_enum_string( k, v );
#include "m.h"
#undef M
}
}
Вы почти всегда хотите защитить от многократного включения. Единственный раз, когда вы не хотите этого делать, это если вы делаете какие-то модные вещи с макросами C, и поэтому вы ХОТИТЕ иметь множественное включение, чтобы получить желаемую генерацию кода (у вас нет примера этого от руки).