在使用GCC 4.7.2和Clang 3.1编译一些C++ 11代码时,我碰到了一个问题,Clang没有设法推导出模板参数GCC成功。 在更抽象的形式,代码看起来是这样的:作为模板参数的可变参数模板:演绎适用于GCC,但不适用于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;
}
通过阅读在§14.3.3实施例和规范的标准的§14.8.2我认为扣除应该有效,但我不能肯定地说。这是我从大楼得到的输出:
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
为什么结果不同? GCC是否马虎,铿锵笨拙,我的代码是否包含未指定的行为或全部?
我同意你的意见。我在C++ 11最终草稿中看到的所有内容都表明这应该起作用。 14.3.3.3尤其相关。 –
你的例子缺少'typedef int Element;',对吧? – Quuxplusone
不,在代码的开头我定义了一个名为Element的结构体。 – psyill