我看过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);
仍然会崩溃。
答案 0 :(得分:2)
PyModule_AddIntConstant(value, "DEFINED", DEFINED_VALUE);
是正确的(假设DEFINED_VALUE
是( C )long
)。
那,结合最后的链接器错误(以及您正在编写代码然后进行测试,等等......)告诉我链接器无法编写新的 .pyd 文件(包含最新更改 - 包括DEFINED
变量),因为以前启动的 python.exe 进程正在使用该文件导入您的模块。
注意:您可以检查函数返回值([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;
}