Boost tuple: увеличение максимального количества элементов

Документация форматирования кортежей гласит:

Текущая версия поддерживает кортежи с 0-10 элементами. При необходимости верхний предел может быть увеличен до, скажем, несколько десятков элементов.

Однако я не мог найти, где он говорит, как это сделать.

Я бы хотел, чтобы кортеж имел элементы BOOST_MPL_LIMIT_VECTOR_SIZE (по умолчанию 20). Это связано с тем, что я сопоставляю между mpl::vectors и boost::tuples и хотел бы, чтобы все контейнеры имели одинаковое количество элементов.

Ответ 1

Класс кортежа объявляется следующим образом:

// - tuple forward declaration -----------------------------------------------
template <
  class T0 = null_type, class T1 = null_type, class T2 = null_type,
  class T3 = null_type, class T4 = null_type, class T5 = null_type,
  class T6 = null_type, class T7 = null_type, class T8 = null_type,
  class T9 = null_type>
class tuple;

Таким образом, его счетчик параметров шаблона устанавливается в верхний предел 10. Однако в будущем (С++ 0x) можно было бы объявить его следующим образом:

template<class... Params> class tuple;

Итак, я не думаю, что в настоящее время возможно увеличить верхний предел на практике. Это было бы возможно следующим образом:

#define VARIADIC_PARAMS_DEFVAL0(type_name, value)
#define VARIADIC_PARAMS_DEFVAL1(type_name, value)   type_name##0 = value
#define VARIADIC_PARAMS_DEFVAL2(type_name, value)   VARIADIC_PARAMS_DEFVAL1(type_name, value), type_name##1 = value
#define VARIADIC_PARAMS_DEFVAL3(type_name, value)   VARIADIC_PARAMS_DEFVAL2(type_name, value), type_name##2 = value
#define VARIADIC_PARAMS_DEFVAL4(type_name, value)   VARIADIC_PARAMS_DEFVAL3(type_name, value), type_name##3 = value
#define VARIADIC_PARAMS_DEFVAL5(type_name, value)   VARIADIC_PARAMS_DEFVAL4(type_name, value), type_name##4 = value
#define VARIADIC_PARAMS_DEFVAL6(type_name, value)   VARIADIC_PARAMS_DEFVAL5(type_name, value), type_name##5 = value
#define VARIADIC_PARAMS_DEFVAL7(type_name, value)   VARIADIC_PARAMS_DEFVAL6(type_name, value), type_name##6 = value
#define VARIADIC_PARAMS_DEFVAL8(type_name, value)   VARIADIC_PARAMS_DEFVAL7(type_name, value), type_name##7 = value
#define VARIADIC_PARAMS_DEFVAL9(type_name, value)   VARIADIC_PARAMS_DEFVAL8(type_name, value), type_name##8 = value
#define VARIADIC_PARAMS_DEFVAL10(type_name, value)  VARIADIC_PARAMS_DEFVAL9(type_name, value), type_name##9 = value
// ...
#define VARIADIC_PARAMS_DEFVAL100(type_name, value) VARIADIC_PARAMS_DEFVAL99(type_name, value), type_name##99 = value
#define VARIADIC_PARAMS_DEFVAL(type_name, value, n) VARIADIC_PARAMS_DEFVAL##n(type_name, value)

#define VARIADIC_MACRO_INVOKE0(macro)    macro(0)
#define VARIADIC_MACRO_INVOKE1(macro)    VARIADIC_MACRO_INVOKE0(macro); macro(1)
#define VARIADIC_MACRO_INVOKE2(macro)    VARIADIC_MACRO_INVOKE1(macro); macro(2)
#define VARIADIC_MACRO_INVOKE3(macro)    VARIADIC_MACRO_INVOKE2(macro); macro(3)
#define VARIADIC_MACRO_INVOKE4(macro)    VARIADIC_MACRO_INVOKE3(macro); macro(4)
#define VARIADIC_MACRO_INVOKE5(macro)    VARIADIC_MACRO_INVOKE4(macro); macro(5)
#define VARIADIC_MACRO_INVOKE6(macro)    VARIADIC_MACRO_INVOKE5(macro); macro(6)
#define VARIADIC_MACRO_INVOKE7(macro)    VARIADIC_MACRO_INVOKE6(macro); macro(7)
#define VARIADIC_MACRO_INVOKE8(macro)    VARIADIC_MACRO_INVOKE7(macro); macro(8)
#define VARIADIC_MACRO_INVOKE9(macro)    VARIADIC_MACRO_INVOKE8(macro); macro(9)
#define VARIADIC_MACRO_INVOKE10(macro)   VARIADIC_MACRO_INVOKE9(macro); macro(10)
// ...
#define VARIADIC_MACRO_INVOKE100(macro)  VARIADIC_MACRO_INVOKE99(macro); macro(100)
#define VARIADIC_MACRO_INVOKE(macro, n)  VARIADIC_MACRO_INVOKE##n(macro)

#define TUPPLE_UPPER_LIMIT      50

#define FORWARD_DECLARE_TUPPLE(N)                       \
template<VARIADIC_PARAMS_DEFVAL(class, null_type, N)    \
class tuple;

FS_VARIADIC_MACRO_INVOKE(FORWARD_DECLARE_TUPPLE, TUPPLE_UPPER_LIMIT);

Ответ 2

Хорошие новости. Найдите ответ, просто используйте макрос, чтобы переопределить максимальные параметры. Библиотека FUSION в boost переопределяет кортеж. Выполните следующие шаги, вы можете легко расширить параметры кортежа

  • определите FUSION_MAX_VECTOR_SIZE перед включением любого файла кортежа
  • включить файл заголовка файла слияния или TR1 версии вместо обычного заголовка кортежа

     #define FUSION_MAX_VECTOR_SIZE 50
     #include <boost/tr1/tuple.hpp>
    

Чтобы лучше понять приведенный выше код, вы можете обратиться к файлу заголовка "boost/tr1/tuple.hpp"

в файле, он имеет другой "BOOST_TR1_USE_OLD_TUPLE", чтобы ссылаться на реализацию старого кортежа.

в реализации фьюжн-кортежа "boost/fusion/tuple/tuple.hpp" есть еще один макрос. "BOOST_FUSION_DONT_USE_PREPROCESSED_FILES". если он не определен, библиотека будет использовать предварительно созданный файл заголовка, максимальный параметр - 50. Если вам нужно больше, я считаю, что вы можете просто определить этот макрос в true. Из кода должно быть хорошо, чтобы иметь больше параметров, хотя я действительно не пробовал. потому что мне 50 намного больше;)

найдите другую проблему, если вы просто определяете FUSION_MAX_VECTOR_SIZE, и вам нужны параметры более 50. Вы должны выдать свой собственный заголовочный файл для векторного шаблона вместо использования существующего файла заголовка процесса. помимо следующего кода, вам также необходимо определить макрос "BOOST_FUSION_DONT_USE_PREPROCESSED_FILES", чтобы исключить прообразанный файл заголовка

#if (FUSION_MAX_VECTOR_SIZE > 50)

#include <boost/fusion/container/vector/vector50.hpp>

namespace boost 
{ 
namespace mpl 
{
#define BOOST_PP_ITERATION_PARAMS_1 (3,(51, FUSION_MAX_VECTOR_SIZE, <boost/mpl/vector/aux_/numbered.hpp>))
#include BOOST_PP_ITERATE()
}

namespace fusion
{
    struct vector_tag;
    struct fusion_sequence_tag;
    struct random_access_traversal_tag;

    // expand vector51 to max
    #define BOOST_PP_FILENAME_1 <boost/fusion/container/vector/detail/vector_n.hpp>
    #define BOOST_PP_ITERATION_LIMITS (51, FUSION_MAX_VECTOR_SIZE)
    #include BOOST_PP_ITERATE()
}
}