44

パラメータパックの内容を印刷するために再帰的な可変個引数テンプレートを作成するにはどうすればよいですか? 私はこれを試していますが、コンパイルに失敗します:

template <typename First, typename ...Args>
std::string type_name () {
    return std::string(typeid(First).name()) + " " + type_name<Args...>();
}
std::string type_name () {
    return "";
}

再帰を終了するにはどうすればよいですか?

4

5 に答える 5

49

実際には、再帰を終了するための非常にエレガントな方法があります:

template <typename Last>
std::string type_name () {
    return std::string(typeid(Last).name());
}

template <typename First, typename Second, typename ...Rest>
std::string type_name () {
    return std::string(typeid(First).name()) + " " + type_name<Second, Rest...>();
}

私は最初に試しtemplate <typename Last>てみtemplate <typename First, typename ...Rest>ましたが、それはあいまいであると見なされました(残りはゼロ要素にすることができます)。この質問は、決定的な解決策を示してくれました:再帰的な可変個引数テンプレート関数のコンパイルエラー


コードの重複を避けるために、次のようにすることもできます。

template <typename Last>
std::string type_name () {
    return std::string(typeid(Last).name());
}

template <typename First, typename Second, typename ...Rest>
std::string type_name () {
    return type_name<First>() + " " + type_name<Second, Rest...>();
}
于 2015-07-11T11:28:31.243 に答える
40

再帰を終了するには部分特殊化を使用する必要がありますが、C++ では自由関数を部分的に特殊化することはできないため、静的メンバー関数を使用して実装クラスを作成する必要があります。

template <typename... Args>
struct Impl;

template <typename First, typename... Args>
struct Impl<First, Args...>
{
  static std::string name()
  {
    return std::string(typeid(First).name()) + " " + Impl<Args...>::name();
  }
};

template <>
struct Impl<>
{
  static std::string name()
  {
    return "";
  }
};

template <typename... Args>
std::string type_name()
{
    return Impl<Args...>::name();
}

int main()
{
  std::cout << type_name<int, bool, char, double>() << std::endl; // "i b c d"
  return 0;
}

の最初の宣言は、g++ 4.6 (およびそれ以下)の欠点に対するImpl単なる回避策です。可変個引数テンプレートを正しく実装すれば、必要ありません。

ideone.com で動作を確認してください

于 2011-08-19T22:19:57.313 に答える
12

存在しない関数の部分的な特殊化の代わりに、型指定子クラスでオーバーロードを使用できます。

#include <string>
#include <iostream>
#include <typeinfo>

template <unsigned int N> struct NumberToType { };

template <typename T>
std::string my_type_name(NumberToType<0> = NumberToType<0>())
{
  return std::string(typeid(T).name());
}

template <typename T, typename ...Args>
std::string my_type_name(NumberToType<sizeof...(Args)> = NumberToType<sizeof...(Args)>())
{
  return std::string(typeid(T).name()) + " " + my_type_name<Args...>(NumberToType<sizeof...(Args)-1>());
}

int main()
{
  std::cout << my_type_name<int, double, char>() << std::endl;
}
于 2011-08-19T23:37:15.367 に答える
7

別の方法として、次の例のように、パラメーター パックをインプレースで展開できます。

#include<string>
#include<iostream>
#include<typeinfo>

template <typename T, typename ...Args>
std::string type_name () {
    std::string str = typeid(T).name();
    int arr[] = { 0, (str += std::string{" "} + typeid(Args).name(), 0)... };
    (void)arr;
    return str;
}

int main() {
    auto str = type_name<int, double, char>();
    std::cout << str << std::endl;
}

実際にそれを行うために再帰は必要ありません。

于 2016-08-26T16:40:47.173 に答える