Модуль .so не импортирует в python: динамический модуль не определяет функцию инициализации - PullRequest
7 голосов
/ 06 ноября 2010

Я пытаюсь написать оболочку Python для функции C. После написания всего кода и его компиляции Python не может импортировать модуль. Я следую примеру, приведенному здесь . Я воспроизвожу это здесь, после исправления некоторых опечаток. Есть файл myModule.c:

#include <Python.h>

/*
 * Function to be called from Python
 */
static PyObject* py_myFunction(PyObject* self, PyObject* args)
{
    char *s = "Hello from C!";
    return Py_BuildValue("s", s);
}
/*
 * Bind Python function names to our C functions
 */
static PyMethodDef myModule_methods[] = {
    {"myFunction", py_myFunction, METH_VARARGS},
    {NULL, NULL}
};

/*
 * Python calls this to let us initialize our module
 */
void initmyModule()
{
    (void) Py_InitModule("myModule", myModule_methods);
}

Так как я на Mac с Python для Macports, я компилирую его как

$ g++ -dynamiclib -I/opt/local/Library/Frameworks/Python.framework/Headers -lpython2.6 -o myModule.dylib myModule.c
$ mv myModule.dylib myModule.so

Однако при импорте я получаю сообщение об ошибке.

$ ipython
In[1]: import myModule
---------------------------------------------------------------------------
ImportError                               Traceback (most recent call last)

/Users/.../blahblah/.../<ipython console> in <module>()

ImportError: dynamic module does not define init function (initmyModule)

Почему я не могу его импортировать?

1 Ответ

5 голосов
/ 06 ноября 2010

Поскольку вы используете компилятор C ++, имена функций будут искажены (например, мои g++ искали void initmyModule() в _Z12initmyModulev).Поэтому интерпретатор python не найдет функцию инициализации вашего модуля.

Вам нужно либо использовать простой компилятор C, либо принудительно связывать C по всему модулю с помощью директивы extern "C" :

#ifdef __cplusplus
extern "C" {
#endif 

#include <Python.h>

/*
 * Function to be called from Python
 */
static PyObject* py_myFunction(PyObject* self, PyObject* args)
{
    char *s = "Hello from C!";
    return Py_BuildValue("s", s);
}

/*
 * Bind Python function names to our C functions
 */
static PyMethodDef myModule_methods[] = {
    {"myFunction", py_myFunction, METH_VARARGS},
    {NULL, NULL}
};

/*
 * Python calls this to let us initialize our module
 */
void initmyModule()
{
    (void) Py_InitModule("myModule", myModule_methods);
}

#ifdef __cplusplus
}  // extern "C"
#endif 
...