12

我在 Debian 挤压上使用 gcc 4.4。考虑以下代码。

#include <map>
#include <string>
using std::map;
using std::string;

// Args lets the user specify additional explicit template arguments
template <typename T,
      template <typename T, typename... Args> class C,
      typename... Args>
C<T, Args...> foo()
{
  C<T, Args...> x;
  return x;
}

int main(void)
{
  map<string, int> a = foo<string, map, int>();
}

所以,这里的想法是Tmatches stringCmatchesmap和模板参数包Argsmatches int。我可能有一些语法错误,如果是,请更正。特别是,如果一个人希望第一个模板参数class C匹配T,其余的匹配模板参数包Argstemplate <typename T, typename... Args> class C那么语法是否正确?

这给出了错误

In function 'int main()':
post.cc:18: error: no matching function for call to 'foo()'

这似乎类似于问题Variadic template templates and perfect forwarding。这个问题表明这是一个 gcc 错误,但也许我错误地认为这些问题是关于同一件事的。

请温柔一点。我对可变参数模板的了解还不到 12 小时;我只是想重写一些旧的 C++ 代码以减少重复。我也有一段时间没有做任何 C++ 了。如果有解决方法,请告诉我。谢谢。

编辑:Variadic 模板模板的评论中建议的解决方法和Ise Wisteria的完美转发对我有用,这表明这是同一个错误。当然,我现在(a)想知道这种解决方法有多脆弱,(b)它为什么有效,以及是什么促使 Ise 想到它。虽然我猜只有一诚可以回答最后一点。:-)

4

2 回答 2

3

正如编辑中所讨论的,我的问题似乎与链接问题、可变参数模板模板和完美转发相同的错误。特别是,链接中给出的解决方法也适用于我的情况。修改后的代码如下:

#include <map>
#include <string>
using std::map;
using std::string;

template <typename T,
      template <typename T, typename... Args> class C,
      typename... Args>
struct X
{
  typedef C<T, Args...> type;
};

template <typename T,
      template <typename T, typename... Args> class C,
      typename... Args>
typename X<T, C, Args...>::type foo()
{
  C<T, Args...> x;
  return x;
}

int main(void)
{
  map<string, int> a = foo<string, map, int>();
}
于 2011-12-15T04:21:13.190 回答
0

我不认为可变参数模板参数可以匹配 g++4.4 中的非可变参数,因此您需要使用非可变参数版本重载 foo 函数。

还要记住 map 实际上有两个以上的模板参数,因此也不会匹配新的 foo 函数。

对您的示例的此添加应该澄清它:

#include <map>
#include <string>
using std::map;
using std::string;

// Args lets the user specify additional explicit template arguments
template <typename T,
          template <typename T, typename... Args> class C,
          typename... Args>
C<T, Args...> foo() {
  C<T, Args...> x;
  return x;
}

template<typename T, template<typename, typename> class C, typename Arg>
C<T, Arg> foo() {
  return C<T, Arg>();
}

template<typename T, typename... Args> class A {};

template<typename T, typename Arg> class B {};

int main(void) {
  map<string, int> a = foo<string, map, int>(); // fails.
  A<string, int> x = foo<string, A, int>();
  B<string, int> y = foo<string, B, int>();
}
于 2011-12-15T03:32:32.223 回答