从C扩展API的python回调中访问全局变量

时间:2014-08-26 09:38:17

标签: python callback python-c-extension python-extensions

我是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(&notify_t, NULL, (void *) &notify_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

1 个答案:

答案 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实例。