【问题标题】:Can't pass arguments to Python function while embedding Python in C++ [closed]在 C++ 中嵌入 Python 时无法将参数传递给 Python 函数 [关闭]
【发布时间】:2021-05-28 04:11:27
【问题描述】:

我在我的 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);

get_list 函数中添加打印语句时,我从 Python 方面注意到的是,参数没有传递给函数。 我想我的语法可能不对?

更新: 我最初使用以下语法尝试了@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):

非常感谢任何帮助。

【问题讨论】:

  • “参数没有被传递给函数” - 那么,传递了什么?这真的需要一个minimal reproducible example - 我们强调“可重现”而不是滴灌小sn-ps
  • 你找到解决这个问题的方法了吗?
  • @Ruzihm 你的答案也适用于我(PyObject_CallMethodObjArgs 也适用于我的问题形式)。问题显然出在 OP 未显示的代码中。
  • @Ruzihm,正如你们提到的那样,问题出在其他地方,我在代码中进行的所有检查都没有发现问题。我建议的您的方法和其他方法(使用 CallMethodObjArgs)现在都可以正常工作。如果可以,请将其他方法添加到您的答案中,我会接受您的。

标签: python c++ python-c-api


【解决方案1】:

应该只有一个格式字符串(可能为 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);
    */
}

(基于this answer's代码)

输出

$ g++ foobar.cpp -lpython3.6m;./a.out
10
15.12

【讨论】:

  • 非常感谢。尝试了你的建议,但没有奏效。我看到方法get_list 被调用(在那里有一个打印到文件的语句)。然而,没有一个参数得到一个传递给它的值。
  • 顺便说一句,这里是get_list 定义行:def get_list(self, num_units, maxTime):
  • @Amir 为我工作
  • 您的建议确实有效。为了完整起见,请使用 CallMethodObjArgs 更新您的答案,我会接受您的答案。再次感谢您的帮助。
  • @Amir 很高兴我能帮上忙。我在答案中包含了您的代码
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2021-11-11
  • 2015-01-23
相关资源
最近更新 更多