Напротив std :: bind, добавьте фиктивные параметры в функцию - PullRequest
11 голосов
/ 18 декабря 2011

Мне нужно нечто противоположное std :: bind, которое добавляет фиктивные параметры к сигнатуре функции вместо того, как boost :: bind связывает параметры.

например. У меня есть эта функция:

std::function<void (void)> myFunc;

Но я хочу преобразовать его в std::function<void(int)> для перехода в эту функцию

void processFunction( std::function<void(int)> func);

Ответы [ 2 ]

6 голосов
/ 18 декабря 2011

Редактировать О, я упомянул очевидное в чате:

@ EthanSteinberg: лямбды?

[] (int realparam, int dummy) { return foo(realparam); }

Но он был отклонен, вот почему я прыгаю на:

Редактировать Я только что понял гораздо более простой подход: http://ideone.com/pPWZk

#include <iostream>
#include <functional>
using namespace std::placeholders;

int foo(int i)
{
    return i*2;
}

int main(int argc, const char *argv[])
{
    std::function<int(int, int)> barfunc = std::bind(foo, (_1, _2));
    std::cout << barfunc(-999, 21) << std::endl;

    // or even (thanks Xeo)
    barfunc = std::bind(foo, _2);
    std::cout << barfunc(-999, 21) << std::endl;
}

Вариативные шаблоны http://ideone.com/8KIsW

Несколько более длинный ответ, основанный на вариационных шаблонах, может привести к уменьшению кода на сайте вызовов (если вы хотите обернуть функции длинным списком аргументов).

#include <iostream>
#include <functional>

int foo(int i)
{
    return i*2;
}

template <typename Ax, typename R, typename... A>
struct Wrap
{
    typedef R (*F)(A...);
    typedef std::function<R(A...)> Ftor;

    Wrap(F f) : _f(f) { }
    Wrap(const Ftor& f) : _f(f) { }

    R operator()(Ax extra, A... a) const
    { return _f(a...); /*just forward*/ }

    Ftor _f;
};

template <typename Ax=int, typename R, typename... A>
std::function<R(Ax, A...)> wrap(R (f)(A...))
{
    return Wrap<Ax,R,A...>(f);
}

template <typename Ax=int, typename R, typename... A>
std::function<R(Ax, A...)> wrap(std::function<R(A...)> functor)
{
    return Wrap<Ax,R,A...>(functor);
}

int main(int argc, const char *argv[])
{
    auto bar = wrap(foo);
    std::function<int(int, int)> barfunc = wrap(foo);

    std::cout << barfunc(-999, 21) << std::endl;

    // wrap the barfunc?
    auto rewrap = wrap(barfunc);
    std::cout << rewrap(-999, -999, 21) << std::endl;

    return 0;
}

Обобщение этого потребует более тяжелой работы. Я думаю, что в прошлых помощниках я видел, как «анализировать» (используя метапрограммирование) сигнатуру std :: function <>, и вы должны иметь возможность заставить его распознавать не пустые функции и, возможно, даже добавлять параметр в конце или в середине (хитро, насколько я могу судить сейчас).

Но для вашего простого случая из OP, похоже, что вы охвачены

3 голосов
/ 18 декабря 2011

Вы можете использовать лямбду, если ваша реализация поддерживает это:

processFunction([=](int a){ myFunc(); });
...