代码之家  ›  专栏  ›  技术社区  ›  Xantium

在扩展中使用pymodule_addintconstant()

  •  0
  • Xantium  · 技术社区  · 6 年前

    我见过 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);
    

    但这又回来了:

    链接:致命错误LNK1104:无法打开文件'build\lib.win32-3.6\winutils.cp36-win32.pyd' 错误:命令“c:\程序文件(x86)\ Microsoft Visual Studio\2017\wdexpress\vc\tools\msvc\14.14.26428\bin\hostx86\x86\link.exe”失败,退出状态为1104

    如何添加 定义 重视我的扩展(以便我可以运行 utilities.DEFINED )?

    编辑:

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

    1 回复  |  直到 6 年前
        1
  •  2
  •   CristiFati    6 年前

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

    再加上最后的链接器错误(以及您正在编写代码,然后进行测试,等等…)告诉我链接器无法编写新的 PYD 文件(包含最新更改-包括 DEFINED 变量),因为它正由以前启动的 巨蜥 导入模块的进程。

    • 关闭 每一个 运行导入模块的解释器(以“解锁” winutils.cp36-win32.pyd文件 )
    • 生成(这次,链接器将能够覆盖文件)
    • 测试(运行 蟒蛇 代码)

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

    @编辑0 (关于2 问题):

    正如我在我的一个评论中指定的,使用 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;
    }