在扩展中使用PyModule_AddIntConstant()

时间:2018-06-13 22:48:00

标签: python c python-3.x python-c-api

我看过Adding symbolic constants with hex values to Python extension module,我正在尝试重现这种效果:

#include <Python.h>
#include <Windows.h>

static PyObject * sys_shutdown(PyObject *self, PyObject *args) {
    int val;

    if (!PyArg_ParseTuple(args, "i", &val))
        val = SHTDN_REASON_MINOR_OTHER; // Provide failsafe

    ExitWindowsEx(EWX_POWEROFF, val); // Shutdown
    return Py_BuildValue("");
}

static PyObject * sys_restart(PyObject *self, PyObject *args) {
    int val;

    if (!PyArg_ParseTuple(args, "i", &val))
        val = SHTDN_REASON_MINOR_OTHER; // Provide failsafe
    ExitWindowsEx(EWX_REBOOT, val); // Restart
    return Py_BuildValue("");
}

static PyObject * sys_log_out(PyObject *self, PyObject *args) {
    int val;

    if (!PyArg_ParseTuple(args, "i", &val))
        val = SHTDN_REASON_MINOR_OTHER; // Provide failsafe

    ExitWindowsEx(EWX_LOGOFF, val); // Log out
    return Py_BuildValue("");
}

static PyMethodDef localMethods[] = {
    {"shutdown", (PyCFunction)sys_shutdown, METH_VARARGS, "..."},
    {"restart", (PyCFunction)sys_restart, METH_VARARGS, "..."},
    {"log_out", (PyCFunction)sys_log_out, METH_VARARGS, "..."},
    {NULL, NULL, 0, NULL}
};

static struct PyModuleDef func = {
    PyModuleDef_HEAD_INIT,
    "utilities",
    "...",
    -1,
    localMethods,
};

PyMODINIT_FUNC PyInit_utilities(void) {
    PyObject *value;

    value = PyModule_New(&func);

    PyModule_AddIntConstant(value, "DEFINED", AF_INET);

    return PyModule_Create(&func);
}

设置脚本:

from distutils.core import setup, Extension

module = Extension(
    "utilities", 
    sources = ["main.c"],
        libraries = ["user32"]
)

setup (
    name = "Utilities",
    version = "1.0",
    ext_modules = [module])

所有内容都按预期构建,但我无法在我的扩展程序中使用DEFINED

import utilities
for i in utilities.__dict__: print(i)
utilities.DEFINED # AttributeError: module 'utilities' has no attribute 'DEFINED'

返回:

__name__
__doc__
__package__
__loader__
__spec__
shutdown
restart
log_out
__file__

我想要像这样返回value

return PyModule_Create(&value);

但是回归:

  

LINK:致命错误LNK1104:无法打开文件'build \ lib.win32-3.6 \ WinUtils.cp36-win32.pyd'   错误:命令'C:\ Program Files(x86)\ Microsoft Visual Studio \ 2017 \ WDExpress \ VC \ Tools \ MSVC \ 14.14.26428 \ bin \ HostX86 \ x86 \ link.exe'失败,退出状态为1104

如何将DEFINED值添加到我的扩展程序中(以便我可以运行utilities.DEFINED)?

修改

如下面的答案所述,关闭所有内容并再次尝试成功构建扩展程序,但使用return PyModule_Create(&value);仍然会崩溃。

1 个答案:

答案 0 :(得分:2)

PyModule_AddIntConstant(value, "DEFINED", DEFINED_VALUE);是正确的(假设DEFINED_VALUE是( C long)。

那,结合最后的链接器错误(以及您正在编写代码然后进行测试,等等......)告诉我链接器无法编写新的 .pyd 文件(包含最新更改 - 包括DEFINED变量),因为以前启动的 python.exe 进程正在使用该文件导入您的模块。

  • 关闭导入模块的每个正在运行的解释器(到&#34;解锁&#34; WinUtils.cp36-win32.pyd
  • 构建(这次,链接器将能够覆盖文件)
  • 测试(运行 Python 代码)

注意:您可以检查函数返回值([Python]: int PyModule_AddIntConstant(PyObject *module, const char *name, long value))。

@ EDIT0 (关于2 nd 问题):

正如我在我的一条评论中指出的那样,使用PyModule_Create(根据[Python]: PyObject* PyModule_New(const char *name),您正在获取未定义的行为):

PyMODINIT_FUNC PyInit_utilities() {
    PyObject *mod = PyModule_Create(&func);
    PyModule_AddIntConstant(mod, "DEFINED", AF_INET);
    return mod;
}