Вызов функции с параметрами, извлеченными из строки - PullRequest
7 голосов
/ 12 декабря 2011

Я смотрю на следующую проблему:

Я получаю строки, отформатированные так:

functionname_parameter1_parameter2_parameter3
otherfunctionname_parameter1_parameter2
.
.
.

, и я хотел бы вызвать функцию с заданными параметрами.Допустим, у меня есть функциональный тест:

void test(int x, float y, std::string z) {}

, и я получаю сообщение:

test_5_2.0_abc

, тогда я хотел бы, чтобы функциональный тест автоматически вызывался следующим образом:

test(5, 2.0, "abc");

Есть ли у вас какие-либо советы о том, как сделать это в C ++?

Ответы [ 5 ]

20 голосов
/ 12 декабря 2011

Обновление: Обновлено stream_function для исправления проблемы порядка оценки аргументов @Nawaz, упомянутой в комментариях, а также удалено std::function для повышения эффективности. Обратите внимание, что исправление порядка оценки работает только для Clang, поскольку GCC здесь не следует стандарту. Пример для GCC с принудительным выполнением заказа можно найти здесь .


Это обычно не так просто сделать. Однажды я написал небольшой класс-оболочку std::function, который извлекает аргументы из std::istream. Вот пример использования C ++ 11:

#include <map>
#include <string>
#include <iostream>
#include <sstream>
#include <functional>
#include <stdexcept>
#include <type_traits>

// for proper evaluation of the stream extraction to the arguments
template<class R>
struct invoker{
  R result;
  template<class F, class... Args>
  invoker(F&& f, Args&&... args)
    : result(f(std::forward<Args>(args)...)) {}
};

template<>
struct invoker<void>{
  template<class F, class... Args>
  invoker(F&& f, Args&&... args)
  { f(std::forward<Args>(args)...); }
};

template<class F, class Sig>
struct stream_function_;

template<class F, class R, class... Args>
struct stream_function_<F, R(Args...)>{
  stream_function_(F f)
    : _f(f) {}

  void operator()(std::istream& args, std::string* out_opt) const{
    call(args, out_opt, std::is_void<R>());
  }

private:  
  template<class T>
  static T get(std::istream& args){
    T t; // must be default constructible
    if(!(args >> t)){
      args.clear();
      throw std::invalid_argument("invalid argument to stream_function");
    }
    return t;
  }

  // void return
  void call(std::istream& args, std::string*, std::true_type) const{
    invoker<void>{_f, get<Args>(args)...};
  }

  // non-void return
  void call(std::istream& args, std::string* out_opt, std::false_type) const{
    if(!out_opt) // no return wanted, redirect
      return call(args, nullptr, std::true_type());

    std::stringstream conv;
    if(!(conv << invoker<R>{_f, get<Args>(args)...}.result))
      throw std::runtime_error("bad return in stream_function");
    *out_opt = conv.str();
  }

  F _f;
};

template<class Sig, class F>
stream_function_<F, Sig> stream_function(F f){ return {f}; }

typedef std::function<void(std::istream&, std::string*)> func_type;
typedef std::map<std::string, func_type> dict_type;

void print(){
  std::cout << "print()\n";
}

int add(int a, int b){
  return a + b;
}

int sub(int a, int b){
  return a - b;
}

int main(){
  dict_type func_dict;
  func_dict["print"] = stream_function<void()>(print);
  func_dict["add"] = stream_function<int(int,int)>(add);
  func_dict["sub"] = stream_function<int(int,int)>(sub);

  for(;;){
    std::cout << "Which function should be called?\n";
    std::string tmp;
    std::cin >> tmp;
    auto it = func_dict.find(tmp);
    if(it == func_dict.end()){
      std::cout << "Invalid function '" << tmp << "'\n";
      continue;
    }
    tmp.clear();
    try{
      it->second(std::cin, &tmp);
    }catch(std::exception const& e){
      std::cout << "Error: '" << e.what() << "'\n";
      std::cin.ignore();
      continue;
    }
    std::cout << "Result: " << (tmp.empty()? "none" : tmp) << '\n';
  }
}

Компилируется под Clang 3.3 и работает как положено ( маленький живой пример ).

Which function should be called?
a
Invalid function 'a'
Which function should be called?
add
2
d
Error: 'invalid argument to stream_function'
Which function should be called?
add
2
3
Result: 5
Which function should be called?
add 2 6
Result: 8
Which function should be called?
add 2   
6
Result: 8
Which function should be called?
sub 8 2
Result: 6

Было весело снова взломать этот класс вместе, надеюсь, вам понравится. Обратите внимание, что вам нужно немного изменить код, чтобы он работал для вашего примера, так как потоки ввода-вывода C ++ имеют пробел в качестве разделителя, поэтому вам нужно заменить все подчеркивания в вашем сообщении пробелами. Это должно быть легко сделать, после этого просто создайте std::istringstream из вашего сообщения:

std::istringstream input(message_without_underscores);
// call and pass 'input'
3 голосов
/ 12 декабря 2011

Вы почти не можете, C ++ не имеет никакого отражения функций.

Вопрос в том, насколько близко вы можете подойти. Такой интерфейс довольно правдоподобен, если он подойдет:

string message = "test_5_2.0_abc";
string function_name = up_to_first_underscore(message);
registered_functions[function_name](message);

Где registered_functions - это map<string,std::function<void,string>>, и вы должны явно сделать что-то вроде:

registered_functions["test"] = make_registration(test);

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

make_registration будет тогда довольно сложной шаблонной функцией, которая принимает указатель на функцию в качестве параметра и возвращает объект std::function, который при вызове разбивает строку на куски, проверяет, есть ли там правильное число, преобразует каждое к правильному типу параметра с помощью boost::lexical_cast, и, наконец, вызывает указанную функцию. Он знал бы «правильный тип» из аргумента шаблона до make_registration - чтобы принять произвольно много параметров, это должен быть вариабельный шаблон C ++ 11, но вы можете подделать его с помощью:

std::function<void,string> make_registration(void(*fn)(void));
template <typename T>
std::function<void,string> make_registration(void(*fn)(T));
template <typename T, U>
std::function<void,string> make_registration(void(*fn)(T, U));
// etc...

Работа с перегрузками и необязательными параметрами добавила бы дополнительные сложности.

Хотя я ничего о них не знаю, я ожидаю, что существуют платформы поддержки C ++ для SOAP или других протоколов RPC, которые могут содержать некоторый соответствующий код.

1 голос
/ 12 декабря 2011

То, что вы ищете, это отражение.И это невозможно в C ++.C ++ разработан с учетом скорости.Если вам требуется проверить библиотеку или код, а затем определить типы в ней и вызвать методы, связанные с этими типами (обычно классами), то я боюсь, что это невозможно в C ++.

Для получения дополнительной справки вы можете обратиться к этой теме.

Как добавить отражение в приложение C ++?

http://en.wikibooks.org/wiki/C%2B%2B_Programming/RTTI

Почему в C ++ нет отражения?

0 голосов
/ 29 января 2014

Я изменил код @ Xeo для правильной работы с gcc, поэтому он обеспечивает правильное расположение параметров. Я только публикую это, так как мне потребовалось некоторое время, чтобы понять оригинальный код и склеить в исполнении заказа. Полный кредит должен все еще идти в @Xeo. Если я обнаружу, что что-то не так с моей реализацией, я вернусь и отредактирую, но до сих пор в своем тестировании я не видел проблем.

#include <map>
#include <string>
#include <iostream>
#include <sstream>
#include <functional>
#include <stdexcept>
#include <type_traits>
#include <tuple>


template<class...> struct types{};

// for proper evaluation of the stream extraction to the arguments
template<class ReturnType>
struct invoker {
    ReturnType result;
    template<class Function, class... Args>
    invoker(Function&& f, Args&&... args) {
        result = f(std::forward<Args>(args)...);
    }
};

template<>
struct invoker<void> {
    template<class Function, class... Args>
    invoker(Function&& f, Args&&... args) {
        f(std::forward<Args>(args)...);
    }
};

template<class Function, class Sig>
struct StreamFunction;

template<class Function, class ReturnType, class... Args>
struct StreamFunction<Function, ReturnType(Args...)> 
{
    StreamFunction(Function f)
        : _f(f) {}

    void operator()(std::istream& args, std::string* out_opt) const 
    {
        call(args, out_opt, std::is_void<ReturnType>());
    }

private:
    template<class T>
    static T get(std::istream& args) 
    {
        T t; // must be default constructible
        if(!(args >> t)) 
        {
            args.clear();
            throw std::invalid_argument("invalid argument to stream_function");
        }
        return t;
    }

    //must be mutable due to const of the class
    mutable std::istream* _args;

    // void return
    void call(std::istream& args, std::string*, std::true_type) const 
    {
        _args = &args;
        _voidcall(types<Args...>{});
    }

    template<class Head, class... Tail, class... Collected>
    void _voidcall(types<Head, Tail...>, Collected... c) const
    {
        _voidcall<Tail...>(types<Tail...>{}, c..., get<Head>(*_args));
    }

    template<class... Collected>
    void _voidcall(types<>, Collected... c) const
    {
        invoker<void> {_f, c...};
    }

    // non-void return
    void call(std::istream& args, std::string* out_opt, std::false_type) const {
        if(!out_opt) // no return wanted, redirect
            return call(args, nullptr, std::true_type());

        _args = &args;
        std::stringstream conv;
        if(!(conv << _call(types<Args...>{})))
            throw std::runtime_error("bad return in stream_function");
        *out_opt = conv.str();
    }

    template<class Head, class... Tail, class... Collected>
    ReturnType _call(types<Head, Tail...>, Collected... c) const
    {
        return _call<Tail...>(types<Tail...>{}, c..., get<Head>(*_args));
    }

    template<class... Collected>
    ReturnType _call(types<>, Collected... c) const
    {
        return invoker<ReturnType> {_f, c...} .result;
    }    


    Function _f;
};

template<class Sig, class Function>
StreamFunction<Function, Sig> CreateStreamFunction(Function f)
{
    return {f};
}

typedef std::function<void(std::istream&, std::string*)> StreamFunctionCallType;
typedef std::map<std::string, StreamFunctionCallType> StreamFunctionDictionary;

Это также работает с Visual Studio 2013, не пробовал более ранние версии.

0 голосов
/ 12 декабря 2011

Вы можете без проблем проанализировать строку, разделить аргументы и отправить их в функцию, но вы не можете сделать это, сославшись на функцию с ее именем в строке, потому что у функции больше нет имени во время выполнения .
Вы можете иметь цепочку if-else if, которая проверяет имя функции, а затем анализирует аргументы и вызывает конкретную функцию.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...