我是python和C-extensions的新手。我正在编写一个python代码,我创建了两个线程,并定义了一个回调函数py_cb()。 在一个线程中,我在某些时间间隔后附加到全局列表,而从另一个线程,我调用C扩展库api。 C-extension api生成一个调用我原始文件中定义的python回调函数(py_cb)的线程。在回调函数中,我试图显示全局列表,但它似乎有不同的id。我使用id(listName)检查了它。有没有办法在C-extension python回调函数中使用正确的全局变量?我假设全局值将在所有线程中共享,是不是这样?另外,请提出更好的解决方案。
以下是C代码:
char *MODULE_NAME = "simple";
---
PyMODINIT_FUNC init_cmodule(void)
{
PyObject *m = Py_InitModule3("_cmodule",module_methods,module_docstring);
if (m == NULL)
return;
}
static PyObject *cmodule_cmodule(PyObject *self, PyObject *args)
{
int value = register_cb();
PyObject *ret = Py_BuildValue("i",value);
return ret;
}
void notifyFooHandle()
{
printf("inside the notify_foo function\n");
pModule = PyImport_ImportModule(MODULE_NAME);
if (!pModule) {
printf ("Failed to load the module\n");
return;
}
PyObject *pFunc;
pFunc = PyObject_GetAttrString(pModule, "py_cb");
if (pFunc && PyCallable_Check(pFunc)) {
PyObject_CallObject(pFunc,NULL);
}
else {
Py_DECREF(pFunc);
PyErr_Print();
printf("Failed to send notification\n");
return;
}
return;
}
void notify_foo(void)
{
int t = 5;
while (t < 10) {
notifyFooHandle();
sleep(5);
t++;
}
return;
}
int register_cb(void)
{
pthread_t notify_t;
int rc = pthread_create(¬ify_t, NULL, (void *) ¬ify_foo,NULL);
---
}
以下是simple.py文件中的python回调API:
def py_cb():
print "Received a call back from cmodule"
global myList
print "Global myList is ", id(myList)
print myList
return
答案 0 :(得分:0)
您能告诉我们C库中的代码吗?如果你在那里初始化python VM然后调用py_cb,很容易理解为什么列表不同:你有两个不同的python VM实例。
修改强>
我认为你的问题是你正在使用两个不同的python实例。首先,你在python中有你的主程序。在那个例子中,你有一个全局的“myList”,所以从那里调用的每个函数都将访问“myList”的特定实例。然后,加载C模块。当C模块打开你原来的python模块以加载py_cb时,你使用的是另一个python实例,你将有第二个“myList”。简而言之,您运行了两个不同的python实例,一个是在运行主python脚本时创建的实例,另一个是在C库中创建的,用于调用py_cb。
如果你想共享一个普通的python实例,你必须在C中创建你的实例,让它在你的C模块中全局访问,在那里插入C函数然后运行你的主python函数。当python调用C函数时,你将进入原始地址空间,而不是新的地址空间,当你回调python时,你将始终使用相同的python实例。