आप यहां क्या करना चाहते हैं, इसे पूरा करने के लिए, आपको अपने टेम्प्लेट के तर्कों को टूप्ले में संग्रहीत करना होगा:
std::tuple<Ts...> args;
इसके अलावा, आपको अपने निर्माता को थोड़ा बदलना होगा। विशेष रूप से, आपकी पैरामीटर सूची में सार्वभौमिक संदर्भों की अनुमति देने के args
साथ आरंभ करना std::make_tuple
:
template <typename F, typename... Args>
Action(F&& func, Args&&... args)
: f(std::forward<F>(func)),
args(std::forward<Args>(args)...)
{}
इसके अलावा, आपको इस तरह एक अनुक्रम जनरेटर स्थापित करना होगा:
namespace helper
{
template <int... Is>
struct index {};
template <int N, int... Is>
struct gen_seq : gen_seq<N - 1, N - 1, Is...> {};
template <int... Is>
struct gen_seq<0, Is...> : index<Is...> {};
}
और आप इस तरह के जनरेटर लेने के संदर्भ में अपनी विधि को लागू कर सकते हैं:
template <typename... Args, int... Is>
void func(std::tuple<Args...>& tup, helper::index<Is...>)
{
f(std::get<Is>(tup)...);
}
template <typename... Args>
void func(std::tuple<Args...>& tup)
{
func(tup, helper::gen_seq<sizeof...(Args)>{});
}
void act()
{
func(args);
}
और वह यह! तो अब आपकी कक्षा इस तरह दिखनी चाहिए:
template <typename... Ts>
class Action
{
private:
std::function<void (Ts...)> f;
std::tuple<Ts...> args;
public:
template <typename F, typename... Args>
Action(F&& func, Args&&... args)
: f(std::forward<F>(func)),
args(std::forward<Args>(args)...)
{}
template <typename... Args, int... Is>
void func(std::tuple<Args...>& tup, helper::index<Is...>)
{
f(std::get<Is>(tup)...);
}
template <typename... Args>
void func(std::tuple<Args...>& tup)
{
func(tup, helper::gen_seq<sizeof...(Args)>{});
}
void act()
{
func(args);
}
};
यहाँ पर आपका पूरा कार्यक्रम कोलिरु पर है।
अद्यतन: यहाँ एक सहायक विधि है जिसके द्वारा टेम्पलेट तर्कों के विनिर्देश आवश्यक नहीं हैं:
template <typename F, typename... Args>
Action<Args...> make_action(F&& f, Args&&... args)
{
return Action<Args...>(std::forward<F>(f), std::forward<Args>(args)...);
}
int main()
{
auto add = make_action([] (int a, int b) { std::cout << a + b; }, 2, 3);
add.act();
}
और फिर, यहाँ एक और डेमो है।