Когда шаблон более специализирован, чем другой? "И" / "Или" путаница с логикой.
В 14.8.2.4p10 проекта C++11 написано
Если для каждого рассматриваемого типа данный шаблон, по крайней мере, столь же специализирован для всех типов и более специализирован для некоторого набора типов, а другой шаблон не более специализирован для каких-либо типов или, по крайней мере, не так специализирован для любых типов, то данный Шаблон более специализирован, чем другой шаблон.
Почему существует "или не так специализировано для каких-либо типов"? Насколько я вижу, если у нас есть список типов
T1, T2, T3
U1, U2, U3
И если все Ц хотя бы как специализированные, а некоторые более специализированные. И ни один из Us не является более специализированным, тогда мне кажется, что из этого следует, что множество T в целом более специализировано, чем множество U, логически говоря. Почему тогда упоминается такой запасной вариант, когда никто из нас не настолько специализирован, как соответствующие Ц?
2 ответа
Обновление: теперь это было добавлено как официальный выпуск C++
Я наконец понял, как читать этот абзац. Ниже я маркировал это
Если для каждого рассматриваемого типа данный шаблон является как минимум специализированным для всех типов, и
- более специализированный для некоторого набора типов, а другой шаблон не более специализированный для каких-либо типов, или
- {другой шаблон} по крайней мере не так специализирован для каких-либо типов,
тогда данный шаблон более специализирован, чем другой шаблон.
Таким образом, следующий первый шаблон также более специализирован, чем второй шаблон
template<typename T> void f(T*);
template<typename T> void f(T);
Обратите внимание, что параметр первого шаблона, по крайней мере, столь же специализирован, как и второй шаблон, но не определен как "более специализированный" - этот термин применяется только в случае, когда оба параметра были ссылочными и применяются определенные условия (см. Пункт 9 из 14.8.2.4) . Правила, очевидно, не предназначены для соблюдения каких-либо формальных законов о порядке. Второй шаблон не настолько специализирован, как первый. Это означает, что применяется вторая пуля, а не первая.
Этот ответ основан на неправильном разборе абстрактного синтаксического дерева стандартного абзаца. Группировка условий, принятая в разделе "Назад к стандарту", оказалась не такой, как предполагалось. Предполагаемая группировка - та, которую Йоханнес Шауб показал в своем ответе.
Почему тогда упоминается такой запасной вариант, когда никто из нас не настолько специализирован, как соответствующие Ц?
Я согласен с вами, что вторая часть (фактически, все второе условие) является излишней.
Некоторая справочная лексика:
Давайте повеселимся с логикой и введем 3 фундаментальных отношения между двумя шаблонами для пары соответствующих параметров:
- Более специализированный, чем: для параметров
Ti
а такжеUi
соответственно, один шаблон соответствует другому, но не наоборот. Я укажу это какTi < Ui
; - Не менее специализированный: по параметрам
Ti
а такжеUi
соответственно один шаблон соответствует другому и наоборот. Я укажу это какTi == Ui
; - Специализация-несравненная: по параметрам
Ti
а такжеUi
соответственно, ни один из шаблонов не соответствует другому для конкретного параметра. Я укажу это какT1 ~ U1
,
Например, во фрагменте кода ниже:
template<typename X> struct A { };
template<typename X> struct B { };
template<typename X> void foo(A<X>, X, A<X>) { } // 1
template<typename X> void foo(X, X, B<X>) { } // 2
Для первого параметра (1) более специализирован, чем (<
) (2); для второго параметра (1) одинаково специализируется как (или " так же специализировано, как ", ==
) (2); по третьему параметру (1) специализация несопоставима с (~
) (2).
И давайте теперь определим производное отношение:
- Шаблон (1) по меньшей мере так же специализирован, как и другой шаблон (2) для соответствующих параметров.
Ti
а такжеUi
когда(Ti < Ui)
или же(Ti == Ui)
т.е. когда (1) более специализирован, чем (2), или (1) столь же специализирован, как (2). Следовательно, в приведенном выше примереT1 <= U1
,T2 <= U2
, а такжеU2 <= T2
,
Вернуться к стандарту:
С помощью нескольких скобок приведенная выше цитата становится (A && (B1 || B2)):
[...] для каждого рассматриваемого типа:
( данный шаблон по крайней мере так же специализирован для всех типов и более специализирован для некоторого набора типов )
AND
( другой шаблон не более специализирован для любых типов
OR
по крайней мере, не так специализирован для любых типов )
Даны два шаблона, которые нужно упорядочить относительно соответствующих последовательностей типов параметров T1, ..., Tn
а также U1, ..., Un
, условие (А):
[...] данный шаблон по крайней мере так же специализирован для всех типов и более специализирован для некоторого набора типов [...]
Означает, что для каждого i = 1..n
, Ti <= Ui
и для некоторых j
в 1..n
, он применяет более строгое условие Tj < Uj
, Отбрасывание индекса i
это означает, что для каждого параметра:
(T < U) || (T == U) // (A)
Это условие помещается в логическое соединение ("и") с другим условием (B), которое, в свою очередь, является логическим разделением ("или") двух подусловий (B1) и (B2). Давайте начнем изучение подусловия (B1):
[...] другой шаблон не более специализирован для любых типов [...]
Это означает, что для любого i
никогда не бывает Ui < Ti
, что означает, что либо:
Ti
более специализированный, чемUi
(Ti < Ui
); или жеTi
а такжеUi
одинаково специализированы (Ui == Ti
); или жеTi
а такжеUi
несопоставимы по специализации (Ui ~ Ti
):
Более формально:
!(U < T) <==> (T < U) || (T == U) || (T ~ U) // (B1)
Теперь давайте посмотрим на второе подусловие (B2), которое находится в логической дизъюнкции с (B1):
[...] по крайней мере не столь специализирован для любых типов [...]
Это отрицание U <= T
, что значит:
!(U <= T) <==> !((U == T) || (U < T)) ==> !(U == T) && !(U < T)
Другими словами, T
а также U
не одинаково специализированы, ни U
более специализированный, чем T
, Следовательно, остаются только следующие возможности:
(T < U) || (T ~ U) // (B2)
Теперь очевидно, что (B2) влечет (B1), потому что (B2) является более ограничительным. Следовательно, их дизъюнкт (B) будет совпадать с (B1), а (B2) избыточен:
(T < U) || (T ~ U) || (T == U) // (B)
Но здесь также очевидно, что (A) является более строгим, чем (B), поэтому соединение (A) и (B) эквивалентно (A).
Заключение:
Все условие (B) является избыточным.