Одна из целей пространств имен - разделить код, поступающий из разных источников. Часто думают, чтобы избежать конфликтов имен, но это идет на более глубокий уровень.
Когда библиотека определяет пространство имен, она (обычно) заявляет, что полностью владеет им. Библиотека оставляет за собой право изменять любые и все аспекты всего, что определено в пространстве имен, обычно гарантируя только API publi c. Это распространяется на предварительные объявления. Любые форвардные объявления в пространстве имен являются доменом библиотеки, если не указано иное, поскольку библиотека может захотеть что-то изменить за кулисами. Некоторые библиотеки относятся к этому серьезно и предоставляют заголовки прямого объявления, но без них вам не повезло.
В общем, не ожидайте, что вы сможете пересылать объявление чего-либо, что вы не контролируете. Даже если он работает изначально, он может сломаться при любом обновлении / исправлении библиотеки.
Переходя к std::set
, пространство имен std
зарезервировано для стандартной библиотеки. Стандартная библиотека предоставляет вам возможность объявлять некоторые вещи в пространстве имен std
в ограниченных обстоятельствах, но в остальном это для вас запрещено. К сожалению для вашей цели, стандартная библиотека не требует заголовков прямого объявления. Если вы хотите, чтобы ваш код был кроссплатформенным и / или стабильным после обновления компилятора, вам нужно будет включить полный заголовок. to namespace std
- неопределенное поведение. Дело не в том, что такие вещи гарантированно что-то сломают, а в том, что языковой стандарт не может гарантировать, что именно сломает. (На самом деле, если вы определили класс с именем supercalifragilisticiwillmisspellthis
в namespace std
, вероятно, ничего не взорвется.) Для простоты применяется метка «неопределенное поведение» и реализациям предоставляется максимальная свобода.