Неоднозначная ссылка на пространство имен во встроенном пространстве имен
Предположим, следующий код:
namespace test
{
namespace detail
{
}
inline namespace v1
{
namespace detail
{
void foo()
{
}
}
}
}
int main()
{
test::detail::foo();
}
Как мы видим, этот код компилируется с помощью Clang; не с GCC, однако - GCC жалуется, что ссылка на namespace detail
неоднозначно:
main.cpp:20:11: error: reference to 'detail' is ambiguous
test::detail::foo();
^
main.cpp:4:5: note: candidates are: namespace test::detail { }
{
^
main.cpp:10:9: note: namespace test::v1::detail { }
{
^
Какой компилятор делает правильную вещь здесь?
2 ответа
GCC правильно:
Члены встроенного пространства имен могут использоваться в большинстве случаев так, как если бы они были членами вложенного пространства имен. В частности, внутреннее пространство имен и включающее его пространство имен добавляются к набору связанных пространств имен, используемых в зависимом от аргументов поиске (3.4.2), где бы ни было одно из них, и директива using, которая именует пространство имен, неявно вставляется во вмещающее пространство. пространство имен как для безымянного пространства имен (7.3.1.1). Кроме того, каждый член встроенного пространства имен может впоследствии быть явно создан (14.7.2) или явно специализирован (14.7.3), как если бы он был членом окружающего пространства имен. Наконец, поиск имени во вмещающем пространстве имен с помощью явной квалификации (3.4.3.2) будет включать членов встроенного пространства имен, введенного директивой using, даже если во входящем пространстве имен есть объявления этого имени.
(Это в 7.3.1/8 в старой нумерации n3337)
Я полагаю, вы видите ошибку Clang # 10361.
GCC правильно.
N3797 утверждает, что
и директива using ( 7.3.4), которая именует внутреннее пространство имен, неявно вставляется в окружающее пространство имен, как и для безымянного пространства имен ( 7.3.1.1).
Таким образом, test::detail
не то же самое пространство имен, что и test::v1::detail
так что поиск test::detail
неоднозначно. Стандарт исключительно ясно, что поиск test::detail
должны включать test::v1::detail
в этом разделе есть много цитат, подтверждающих это, но ничего не говорится о том, что они должны рассматриваться как одно и то же пространство имен.
Возможно, я бы сказал, что поведение Clang лучше, но GCC правильно.