python - 在 C++ 中嵌入 Python 时无法将参数传递给 Python 函数
问题描述
我在我的 C++ 代码中嵌入了一个 Python 模块,并且我正在使用 Python/C API。我需要调用 Python 模块函数并获取结果。该函数获取一个无符号整数和一个双精度作为输入参数并输出一个列表。我使用以下方法调用该函数:
unsigned int num_units = 10;
double max_time = 15.12;
PyObject *output_list = PyObject_CallMethod(sample_object, "get_list",
"(I)", num_units, "(d)", max_time);
在函数内部添加打印语句时,我从 Python 方面注意到的get_list
是,参数没有传递给函数。我想我的语法可能不对?
更新:我最初使用以下语法尝试了@Ruzihm 的建议,但没有成功。事实证明,python 代码中还有其他语法问题阻止了代码的正确执行,并且我的错误检查都没有发现它。解决问题后,代码运行完美。
PyObject *output_list = PyObject_CallMethod(sample_object, "get_list",
"Id",num_units, max_time);
我还尝试将 CallMethodObjArgs 与以下代码一起使用,效果也很好:
PyObject *func_obj, *num_units_obj, *max_time_obj;
num_units_obj = PyLong_FromUnsignedLong(num_units);
max_time_obj = PyFloat_FromDouble(max_time);
func_obj = PyUnicode_FromString("get_list");
PyObject *output_list = PyObject_CallMethodObjArgs(sample_object,
func_obj, num_units_obj,
max_time_obj, NULL);
Py_DECREF(func_obj);
Py_DECREF(num_units_obj);
Py_DECREF(max_time_obj);
Python 函数定义也如下所示:
def get_list(self, num_units, max_time):
很感谢任何形式的帮助。
解决方案
应该只有一个格式字符串(可能为 NULL),后跟任何/所有输入。括号用于指定大小为零或一的元组,您没有提到需要。所以,只需这样做:
PyObject *output_list = PyObject_CallMethod(sample_object,
"get_list", "Id", num_units, max_time);
完整示例:
foobar.py
class foobar():
def get_list(self, num_units, max_time):
print(num_units)
print(max_time)
foobar.cpp
#define PY_SSIZE_T_CLEAN
#include "python3.6m/Python.h"
#include <iostream>
int main() {
PyObject *module, *dict, *python_class, *sample_object;
setenv("PYTHONPATH", ".", 1);
Py_Initialize();
module = PyImport_ImportModule("foobar");
if (module == nullptr)
{
std::cout << "Failed to import module.";
return 1;
}
dict = PyModule_GetDict(module);
if (dict == nullptr)
{
std::cout << "Failed to get module dict.";
return 1;
}
Py_DECREF(module);
python_class = PyDict_GetItemString(dict, "foobar");
if (python_class == nullptr)
{
std::cout << "Failed to get class.";
return 1;
}
Py_DECREF(dict);
sample_object = PyObject_CallObject(python_class, nullptr);
if (sample_object == nullptr)
{
std::cout << "Failed to instantiate object.";
return 1;
}
Py_DECREF(python_class);
unsigned int num_units = 10;
double max_time = 15.12;
PyObject *output_list = PyObject_CallMethod(sample_object, "get_list",
"Id", num_units, max_time);
/* Alternatively, use PyObject_CallMethodObjArgs
PyObject *func_obj, *num_units_obj, *max_time_obj;
num_units_obj = PyLong_FromUnsignedLong(num_units);
max_time_obj = PyFloat_FromDouble(max_time);
func_obj = PyUnicode_FromString("get_list");
PyObject *output_list = PyObject_CallMethodObjArgs(sample_object,
func_obj, num_units_obj,
max_time_obj, NULL);
Py_DECREF(func_obj);
Py_DECREF(num_units_obj);
Py_DECREF(max_time_obj);
*/
}
(基于此答案的代码)
输出
$ g++ foobar.cpp -lpython3.6m;./a.out
10
15.12
推荐阅读
- python - 如何写入子目录 Python 2.7 中的文件?
- tensorflow - 验证集上的去噪自动编码器给出了不好的结果,甚至对于训练图像在实例中随机噪声
- python - 如何解决 FileNotFoundError: [Errno 2] No such file or directory for Python 3.7/Mac
- javascript - 以特定格式将 mysql 数据检索到 javascript 的问题
- python - 编写一个算法来计算字典中子字典的数量
- javascript - 在javascript中将IST缩写解析为+0530
- python - urllib.error.HTTPError:HTTP 错误 404:从 Metacritic 抓取数据时未找到 Python
- mysql - 除了“数量”之外,所有列都给出了正确的值 - 使用 MySQL WITH ROLLUP
- react-native - 如何在导航参数中传递获取响应?
- firebase - 将用户导航到 appStore 并首次下载时,DynamicLink 深层链接 url 为零