Вложение Python в С++: ImportError: нет модуля с именем pyfunction

Привет. Я пытаюсь вставить python (2.7) в С++ (g++ 4.8.2) и, следовательно, вызывать функцию python из С++. Это базовый код, предоставляемый в документации по python для встраивания:

Это мой файл call_function.cpp

#include <Python.h>

int main(int argc, char *argv[])
{
    PyObject *pName, *pModule, *pDict, *pFunc;
    PyObject *pArgs, *pValue;
    int i;

    if (argc < 3) {
        fprintf(stderr,"Usage: call pythonfile funcname [args]\n");
        return 1;
    }
   /* char pySearchPath[] = "/usr/include/python2.7";
    Py_SetPythonHome(pySearchPath);*/
    Py_Initialize();
    pName = PyString_FromString(argv[1]);
    /* Error checking of pName left out */

    pModule = PyImport_Import(pName);
    Py_DECREF(pName);

    if (pModule != NULL) {
        pFunc = PyObject_GetAttrString(pModule, argv[2]);
        /* pFunc is a new reference */

        if (pFunc && PyCallable_Check(pFunc)) {
            pArgs = PyTuple_New(argc - 3);
            for (i = 0; i < argc - 3; ++i) {
                pValue = PyInt_FromLong(atoi(argv[i + 3]));
                if (!pValue) {
                    Py_DECREF(pArgs);
                    Py_DECREF(pModule);
                    fprintf(stderr, "Cannot convert argument\n");
                    return 1;
                }
                /* pValue reference stolen here: */
                PyTuple_SetItem(pArgs, i, pValue);
            }
            pValue = PyObject_CallObject(pFunc, pArgs);
            Py_DECREF(pArgs);
            if (pValue != NULL) {
                printf("Result of call: %ld\n", PyInt_AsLong(pValue));
                Py_DECREF(pValue);
            }
            else {
                Py_DECREF(pFunc);
                Py_DECREF(pModule);
                PyErr_Print();
                fprintf(stderr,"Call failed\n");
                return 1;
            }
        }
        else {
            if (PyErr_Occurred())
                PyErr_Print();
            fprintf(stderr, "Cannot find function \"%s\"\n", argv[2]);
        }
        Py_XDECREF(pFunc);
        Py_DECREF(pModule);
    }
    else {
        PyErr_Print();
        fprintf(stderr, "Failed to load \"%s\"\n", argv[1]);
        return 1;
    }
    Py_Finalize();
    return 0;
}

Теперь мой python script сохраняется как pyfunction.py, помещенный в ту же папку, что и call_function.cpp.

Это pyfunction.py:

def multiply(a,b):
    print "Will compute", a, "times", b
    c = 0
    for i in range(0, a):
        c = c + b
    return c

Теперь, используя Терминал, я звоню:

$ g++ call_function.cpp -I/usr/include/python2.7 -lpython2.7 -o call_function

(успешно компилируется без каких-либо ошибок) (Запуск программы)

$ ./call_function pyfunction multiply 2 3

(я получаю эту ОШИБКУ):

ImportError: No module named pyfunction    
Failed to load "pyfunction"

Я не понимаю, как это возможно. Я следил за документацией и все еще получаю ошибку.

Как он не может найти pyfunction.py, когда он помещен в тот же каталог.

Ответ 1

Попробуйте следующее:

 $ PYTHONPATH=. ./call_function pyfunction multiply 2 3

если это не сработает, попробуйте сделать __init__.py файл в этом каталоге и повторите попытку.

UPDATE:

Я думаю, что PYTHONPATH является временным решением для тестирования. Если вы хотите иметь каталог, когда все ваши встроенные модули живут, вы должны включить в свой встроенный интерпретатор что-то равное этому:

import sys
sys.path.insert(0, "./path/to/your/modules/")

Вы можете сделать это, вероятно, на python в вашем интерпретаторе или на уровне C. Это добавит путь поиска так же, как PYTHONPATH, но будет более устойчивым и элегантным (IMHO).

Ответ 2

Поместите следующее в код C/С++ сразу после Py_Initialize();

PyRun_SimpleString("import sys");
PyRun_SimpleString("sys.path.append(\".\")");

Ответ 3

Привет всем, кто сталкивается с той же проблемой, я нашел решение! setenv() - это функция, определенная в stdlib.h, которая задает переменную окружения. Просто нужно запустить его!

setenv("PYTHONPATH",".",1);

для получения дополнительной информации о setenv:

$ man setenv

Все самое лучшее:) Кроме того, благодаря @spinus

Ответ 4

решение, предоставляемое spinus, работает, если файл python не импортирует дополнительную библиотеку python.

Однако, если файл python импортирует дополнительную библиотеку, скажем, numpy, приведенный выше код сработает следующим образом:

:~/programs/python$ ./a.out myModule multiply 4 3
Traceback (most recent call last):
  File "/home/a/programs/python/myModule.py", line 1, in <module>
    import numpy
ImportError: No module named 'numpy'
Failed to load "myModule"

Как замечание, импорт библиотеки python из C не работает:

PyObject *pNumpy = PyUnicode_FromString("numpy");
PyObject *pModuleA = PyImport_Import(pNumpy); 

Кто-нибудь знает, как вызвать из C python-функций, которые зависят от некоторых других библиотек python?

Ответ 5

Вы также можете попытаться включить этот код в свою программу на c

Py_Initialize();
PyObject *sys = PyImport_ImportModule("sys");
PyObject *path = PyObject_GetAttrString(sys, "path");
PyList_Append(path, PyUnicode_FromString("."));

Учитесь у здесь