Durante la compilazione di codice C++ 11 con GCC 4.7.2 e Clang 3.1, ho riscontrato un problema con Clang che non riusciva a dedurre un argomento modello in cui GCC ha successo. In una forma più astratta, il codice simile a questo:Modello Variadic come parametro template: la deduzione funziona con GCC ma non con Clang
src/test.cc:
struct Element {
};
template <typename T>
struct FirstContainer {
};
template <typename T, typename U = Element>
struct SecondContainer {
};
template <template <typename> class Container>
void processOrdinary(Container<Element> /*elements*/) {
}
template <template <typename, typename> class Container>
void processOrdinary(Container<Element, Element> /*elements*/) {
}
template <template <typename, typename...> class Container>
void processVariadic(Container<Element> /*elements*/) {
}
int main() {
// This function instantiation works in both GCC and Clang.
processOrdinary(FirstContainer<Element>{});
// This function instantiation works in both GCC and Clang.
processOrdinary(SecondContainer<Element>{});
// This function instantiation works in both GCC and Clang.
processVariadic(FirstContainer<Element>{});
// This function instantiation works in both GCC and Clang.
processVariadic<SecondContainer>(SecondContainer<Element>{});
// This function instantiation works in GCC but not in Clang.
processVariadic(SecondContainer<Element>{});
return 0;
}
Dalla lettura degli esempi in §14.3.3 e le specifiche in §14.8.2 dello standard Penso che la deduzione dovrebbe funzionare, ma non posso dirlo con certezza. Questo è l'output che ottengo dall'edificio:
mkdir -p build-gcc/
g++ -std=c++0x -W -Wall -Wextra -Weffc++ -pedantic -c -o build-gcc/test.o src/test.cc
g++ -o build-gcc/test build-gcc/test.o
mkdir -p build-clang/
clang++ -std=c++11 -Weverything -Wno-c++98-compat -c -o build-clang/test.o src/test.cc
src/test.cc:34:3: error: no matching function for call to 'processVariadic'
processVariadic(SecondContainer<Element>{});
^~~~~~~~~~~~~~~
src/test.cc:21:6: note: candidate template ignored: failed template argument deduction
void processVariadic(Container<Element> /*elements*/) {
^
1 error generated.
make: *** [build-clang/test.o] Fel 1
Perché i risultati differiscono? È GCC sciatto, Clang stupido, il mio codice contiene un comportamento non specificato o tutti?
Sono d'accordo con te. Tutto ciò che ho visto nella bozza finale del C++ 11 dovrebbe indicare che questo dovrebbe funzionare. 14.3.3.3 è particolarmente rilevante. –
Nell'esempio manca un 'typedef int Element;', giusto? – Quuxplusone
No, all'inizio del codice definisco una struttura con il nome Elemento. – psyill