我试图通过Python在我的游戏引擎中创建一个很好的界面来进行HTTP调用,但是我遇到了一些问题。
我有一个功能get_async
,它会为指定的网址发起curl请求。该函数在http_manager
类中定义,如下所示:
struct http_manager
{
typedef function<void(boost::shared_ptr<http_response>)> response_callback_type;
typedef function<void(size_t)> write_callback_type;
void get_async(
const string& url,
const http_headers_type& headers = http_headers_type(),
const http_data_type& data = http_data_type(),
response_callback_type on_response = nullptr,
write_callback_type on_write = nullptr
);
};
我能够在Python中成功进行此调用:
http.get_async('http://www.google.ca')
但是,我想打这样的电话:
http.get_async('http://www.google.ca', on_response=f)
这里的关键是我想通过名称显式指定参数,让所有其他参数成为默认值,就像常规Python一样。
不幸的是,当我这样做时,我从Python中得到以下错误:
ArgumentError: Python argument types in
HttpManager.get_async(HttpManager, str)
did not match C++ signature:
get_async(struct naga::http_manager {lvalue}, class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > url)
get_async(struct naga::http_manager {lvalue}, class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > url, class std::vector<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > headers)
get_async(struct naga::http_manager {lvalue}, class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > url, class std::vector<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > headers, class std::map<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,struct std::less<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > const ,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > data)
get_async(struct naga::http_manager {lvalue}, class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > url, class std::vector<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > headers, class std::map<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,struct std::less<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > const ,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > data, class std::function<void __cdecl(class boost::shared_ptr<struct naga::http_response>)> on_response)
get_async(struct naga::http_manager {lvalue}, class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > url, class std::vector<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > headers, class std::map<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> >,struct std::less<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > >,class std::allocator<struct std::pair<class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > const ,class std::basic_string<char,struct std::char_traits<char>,class std::allocator<char> > > > > data, class std::function<void __cdecl(class boost::shared_ptr<struct naga::http_response>)> on_response, class std::function<void __cdecl(unsigned int)> on_write)
当我明确传递3个参数(HttpManager.get_async(HttpManager, str)
,self
和{{1}时,我很困惑为什么它认为参数签名是url
}})。
以下是on_response
块中的相关位:
BOOST_PYTHON_MODULE
感谢您阅读,非常感谢任何帮助!
答案 0 :(得分:8)
整体问题是C ++只使用默认参数来丢失尾随参数,而Boost.Python没有值来提供非尾随缺失参数。
当Boost.Python内部无法调度函数时,异常消息仅捕获位置参数。异常消息尝试捕获调用者的意图,但以C ++函数调用格式显示它。由于C ++仅支持位置参数,因此没有明显的格式来显示非位置参数。考虑一下这个简单的example:
#include <boost/python.hpp>
void f() {}
BOOST_PYTHON_MODULE(example)
{
boost::python::def("f", &f);
}
使用以下消息调用example.f(a=0)
抛出Boost.Python.ArgumentError
:
Traceback (most recent call last):
File "<stdin>", line 2, in <module>
Boost.Python.ArgumentError: Python argument types in
example.f()
did not match C++ signature:
f(void)
虽然Boost.Python完全知道调用者提供了一个参数,但异常只捕获了位置参数。在上面的示例中,没有位置参数,因此输出为example.f()
。在原始问题中,调用http.get_async('http://www.google.ca', on_response=f)
导致显示HttpManager.get_async(HttpManager, str)
的异常,因为只提供了两个位置参数:类型HttpManager
和str
对象的隐式自参数。 / p>
在C ++中,默认参数不是函数类型的一部分。此外,在调用具有默认参数的函数时,默认参数仅用于缺少尾随参数的位置。例如,考虑:
void f(int a, int b=1, int c=2);
上述函数的类型为void(int, int, int)
,无法调用函数f
,只为参数a
和c
提供参数。如果希望使用参数f
的非默认值调用c
,则必须提供所有三个参数。
但是,如果Python中声明的函数类似,则可以调用f
仅为参数a
和c
提供参数,因为Python支持关键字参数:
def f(a, b=1, c=2):
pass
f(0, c=42) # Invokes f(0, b=1, c=42)
考虑这个example,其中带有默认参数的C ++函数已作为重载的Python函数提供:
#include <boost/python.hpp>
void f(int a, int b=1, int c=2) {}
BOOST_PYTHON_FUNCTION_OVERLOADS(f_overloads, f, 1, 3)
BOOST_PYTHON_MODULE(example)
{
namespace python = boost::python;
python::def("f", &f, f_overloads(
python::args("a", "b", "c")));
}
Boost.Python被告知C ++函数f
的类型为void(int, int, int)
,并且可以使用至少1个参数调用,最多可以调用3. Boost.Python也不知道默认参数值是什么。因此,Boost.Python创建了重载来支持调用:
f(int a)
f(int a, int b)
f(int, a, int b, int c)
因此,当一个人试图在Python中调用example.f(0, c=42)
时,函数调度将失败,因为当为b
提供值时,Boost.Python需要为c
提供值,并且未指定b
的值。
Traceback (most recent call last):
File "<stdin>", line 2, in <module>
Boost.Python.ArgumentError: Python argument types in
example.f(int)
did not match C++ signature:
f(int a)
f(int a, int b)
f(int a, int b, int c)
这种行为是http.get_async('http://www.google.ca', on_response=f)
失败的原因,因为Boost.Python无法调用:
http_manager.get_async(url, ???, ???, on_response);
继续该示例,如果Python提供了默认参数,则它可以为非尾随参数提供值。在公开函数时,可以通过分配给boost::python::arg
对象来为参数提供默认值。以下example使用默认参数公开函数f
:
#include <boost/python.hpp>
void f(int a, int b=1, int c=2) {}
BOOST_PYTHON_FUNCTION_OVERLOADS(f_overloads, f, 1, 3)
BOOST_PYTHON_MODULE(example)
{
namespace python = boost::python;
python::def("f", &f, f_overloads(
(python::arg("a"),
python::arg("b")=1,
python::arg("c")=2)));
}
Boost.Python知道默认参数,它可以成功调用example.f(0, c=42)
。