2012-12-13 36 views
1

所以我有一个GUI程序,有很多“东西”在继续。我正在添加一个python脚本界面,以便有人可以与此环境进行交互。我使用boost python。所以我的第一件事就是我想创建一个新的模块。为简单起见,现在我只是模块是世界你好...嵌入在C++中的Python

#include <boost/python.hpp>              

char const* greet() {               
    return "hello, world" ;              
}                    

BOOST_PYTHON_MODULE(cerrnimapi) {            
    boost::python::def("greet", greet) ;           
} 

在我的系统我有一个类,看起来像这样...

Controller::Controller() {   
    Py_Initialize() ;                

    main_module = boost::python::import("__main__") ;       
    main_namespace = main_module.attr("__dict__") ;        
}                                      

void Controller::execute_script(std::string filename) {      
    try {                   
    boost::python::api::object ignored =           
     boost::python::exec_file(filename.c_str(), main_namespace) ;    
    } catch(boost::python::error_already_set const &) {       
    if (PyErr_ExceptionMatches(PyExc_ZeroDivisionError)) {      
    } else {                  
     PyErr_Print();               
    }                   
    }                    
} 

现在,当我去执行脚本我得到一个错误的图形用户界面...

Traceback (most recent call last): 
    File "/home/mokon/repository/trunk/python.py", line 1, in <module> 
    import cerrnimapi 
ImportError: No module named cerrnimapi 

所以当然,我构建了一些错误。我的构建系统使用自动工具使这里有涉及到这一点,构建系统的几件...

在configure.ac:

AM_PATH_PYTHON                 
AC_ARG_VAR([PYTHON_INCLUDE], [Include flags for python, bypassing python-config]) 
AC_ARG_VAR([PYTHON_CONFIG], [Path to python-config])        
AS_IF([test -z "$PYTHON_INCLUDE"], [            
    AS_IF([test -z "$PYTHON_CONFIG"], [           
    AC_PATH_PROGS([PYTHON_CONFIG],            
        [python$PYTHON_VERSION-config python-config],     
        [no],               
        [`dirname $PYTHON`])           
    AS_IF([test "$PYTHON_CONFIG" = no], [AC_MSG_ERROR([cannot find python-config for $PYTHON.])]) 
    ])                    
    AC_MSG_CHECKING([python include flags])          
    PYTHON_INCLUDE=`$PYTHON_CONFIG --includes`          
    AC_MSG_RESULT([$PYTHON_INCLUDE])            
])                    

AC_ARG_VAR([PYTHON_LD], [Linker flags for python, bypassing python-config])  
AS_IF([test -z "$PYTHON_LD"], [             
    AS_IF([test -z "$PYTHON_CONFIG"], [           
    AC_PATH_PROGS([PYTHON_CONFIG],            
        [python$PYTHON_VERSION-config python-config],     
        [no],               
        [`dirname $PYTHON`])           
    AS_IF([test "$PYTHON_CONFIG" = no], [AC_MSG_ERROR([cannot find python-config for $PYTHON.])]) 
    ])                    
    AC_MSG_CHECKING([python linker flags])           
    PYTHON_LD=`$PYTHON_CONFIG --ldflags`           
    AC_MSG_RESULT([$PYTHON_LD])             
]) 

在我的OBJ /目录Makefile.am ...

pyexec_LTLIBRARIES = cerrnimapi.la            
cerrnimapi_la_SOURCES = ${SRC_DIR}/lib/PythonAPI.cpp        
cerrnimapi_la_LDFLAGS = -avoid-version -module $(PYTHON_LD)      
cerrnimapi_la_CXXFLAGS = $(PYTHON_INCLUDE) 

我的makefile和我的主程序一起构建了共享库及其obj文件夹。这没有帮助。我也做了一个make install来安装cerrnimapi lib到python文件夹中。这没有帮助。

我也尝试将PythonAPI.cpp添加到我的主程序源中,但无济于事。

任何想法?让我知道什么额外的信息会有帮助。

+0

另外,更多的,我想想我认为我做错了。我不应该只是能够将我的库编译到我的C++程序中,嵌入式python interp将能够看到符号? –

+0

您的构建过程是否生成名为“cerrnimapi.so”的共享库?这就是进口要寻找的东西。至于图书馆中的静态链接,这不是直接鼓励,但如果你想破解它,这里有信息:http://mdqinc.com/blog/2011/08/statically-linking-python-with-cython-generated-模块和 - 包/。 –

回答

1

有些事情要检查:

  • 运行nm在你的.so文件(这可能是在.libs),以确保您的模块初始化func被导出。
  • 使您的程序打印出sys.path(使用PyRun_SimpleString)的值以查看期望模块启动的位置。如果您只为解释器定义模块,则可能不希望将其安装在$pyexecdir中。
  • 阅读Extending Embedded Python文章。你根本不需要构建动态库,除非你正在尝试插件体系结构。

款式的一点:你应该尝试找到$PYTHON_CONFIG外面你的测试对$PYTHON_INCLUDE$PYTHON_LD所以你没有做的AC_PATH_PROGS两次:

AM_PATH_PYTHON                 
AC_ARG_VAR([PYTHON_CONFIG], [Path to python-config])        
AS_IF([test -z "$PYTHON_CONFIG"], [           
    AC_PATH_PROGS([PYTHON_CONFIG],            
       [python$PYTHON_VERSION-config python-config],     
       [no],               
       [`dirname $PYTHON`])           
])                    

AC_ARG_VAR([PYTHON_INCLUDE], [Include flags for python, bypassing python-config]) 
AS_IF([test -z "$PYTHON_INCLUDE"], [            
    AC_MSG_CHECKING([python include flags])          
    AS_IF([test "$PYTHON_CONFIG" = no], [AC_MSG_ERROR([cannot find python-config for $PYTHON.])]) 
    PYTHON_INCLUDE=`$PYTHON_CONFIG --includes`          
    AC_MSG_RESULT([$PYTHON_INCLUDE])            
])                    

AC_ARG_VAR([PYTHON_LD], [Linker flags for python, bypassing python-config])  
AS_IF([test -z "$PYTHON_LD"], [             
    AC_MSG_CHECKING([python linker flags])           
    AS_IF([test "$PYTHON_CONFIG" = no], [AC_MSG_ERROR([cannot find python-config for $PYTHON.])]) 
    PYTHON_LD=`$PYTHON_CONFIG --ldflags`           
    AC_MSG_RESULT([$PYTHON_LD])             
]) 
+0

因此,我创建的python api需要访问我的主应用程序中的模型,因为它需要与它交互。例如,我有一个关闭该系统的简单命令。为此,它抓取一个全局对象并实质上调用它的关闭方法。当然问题是,从.so动态链接的文件调用该文件,因此全局为空。我需要它从主程序访问。我没有看到你所指的那篇文章在哪里 –

+0

任何帮助,将不胜感激。我觉得这是非常基本的...它已经有一段时间了,因为我不得不担心这个... –

+0

我链接到你的手册部分显示了如何在你的主程序中定义一个模块。你根本不需要有.so。要回到你的主程序,看起来你必须传递函数指针:http://stackoverflow.com/questions/3454909/sharing-data-with-a-dynamically-loaded-library-dlopen-dlsym。不要这样做,它会变得混乱。 –