2012-08-15 3 views
9

Si consideri il seguente codice:È possibile generare un template pack variadic da una dimensione e dal suo contenuto?

template<unsigned int... TSIZE> 
struct Base {}; 
template<unsigned int TORDER, unsigned int TDIM> 
struct Derived : public Base</* TDIM, TDIM, ... TDIM (TORDER times) */> {}; 

Pensi che esiste un trucco per generare correttamente i parametri del modello di base sulla seconda riga di questo esempio? Ad esempio voglio che l'Derived<3, 5> erediti da Base<5, 5, 5>. Come farlo ?

risposta

10

Con un po 'di TMP, questo non è così difficile, dopo tutto:

template<unsigned ToGo, class T, T Arg, template<T...> class Target, T... Args> 
struct generate_pack 
    : generate_pack<ToGo-1, T, Arg, Target, Args..., Arg> 
{ // build up the 'Args' pack by appending 'Arg' ... 
}; 

template<class T, T Arg, template<T...> class Target, T... Args> 
struct generate_pack<0, T, Arg, Target, Args...> 
{ // until there are no more appends to do 
    using type = Target<Args...>; 
}; 

template<unsigned Num, class T, T Arg, template<T...> class Target> 
using GeneratePack = typename generate_pack<Num, T, Arg, Target>::type; 

template<unsigned int... TSIZE> 
struct Base{}; 

template<unsigned int TORDER, unsigned int TDIM> 
struct Derived 
    : GeneratePack<TORDER, unsigned, TDIM, Base> 
{ 
}; 

Live example.

+0

Oh, è possibile derivare in modo ricorsivo? Questo è figo. – jrok

+1

si chiama inoltro metafunzione modello da parte degli autori Boost.MPL. – TemplateRex