0

我有一个 pyd pytest.pyd,其中声明了两个函数:say_hello 和 add_numbers。所以我想用 LoadLibraryEx 在 C++ 中动态加载这个 pyd。但是,当我尝试调用 initpytest func 时,它失败了。

const char* funcname = "initpytest";

HINSTANCE hDLL = LoadLibraryEx(L"D:\\msp\\myproj\\Test\\pytest.pyd", NULL, LOAD_WITH_ALTERED_SEARCH_PATH);

FARPROC p = GetProcAddress(hDLL, funcname);

(*p)(); // fail

在输出错误中:致命的 Python 错误:PyThreadState_Get:在 Test.exe 中的 0x00007FFCD0CC286E (ucrtbase.dll) 处没有当前线程未处理的异常:请求致命的程序退出。

这里是生成到 pyd 之前的扩展代码:

#include "Python.h"

static PyObject* say_hello(PyObject* self, PyObject* args)
{
    const char* msg;

    if(!PyArg_ParseTuple(args, "s", &msg))
    {
        return NULL;
    }
    printf("This is C world\nYour message is %s\n", msg);
    return Py_BuildValue("i", 25);
}

static PyObject* add_numbers(PyObject* self, PyObject* args)
{
    double a, b;

    if (!PyArg_ParseTuple(args, "dd", &a, &b))
    {
        return NULL;
    }
    double res = a + b;
    return Py_BuildValue("d", res);
}

static PyMethodDef pytest_methods[] = {
    {"say_hello", say_hello, METH_VARARGS, "Say Hello!"},
    {"add_numbers", add_numbers, METH_VARARGS, "Adding two numbers"},
    {NULL, NULL, 0, NULL}
};

PyMODINIT_FUNC initpytest(void)
{
    Py_InitModule("pytest", pytest_methods);
}
4

1 回答 1

0

在没有适当的最小、可重现的例子的情况下,不可能确定。但是,这可能是因为您尚未初始化解释器:(例如,请参阅此问题)。您需要Py_Initialize在使用任何 Python 函数之前调用。

我是否可以建议您使用普通的 Python C-API 工具来运行模块(而不是自己使用LoadLibraryEx!),直到您完全了解嵌入 Python 的工作原理。您可能会考虑PyImport_AppendInittab(在初始化之前)直接设置您的函数并避免使用 Python 搜索路径。

于 2021-06-07T18:14:54.187 回答