C++ 使用boost.python在cygwin中手动构建python扩展

C++ 使用boost.python在cygwin中手动构建python扩展,c++,build,cygwin,boost-python,python-extensions,C++,Build,Cygwin,Boost Python,Python Extensions,很抱歉这样一个笼统的标题,但我不太确定我到底错过了什么或者我做错了什么。我的目标是在cygwin下使用boost.python构建一个python扩展,并避免使用boost.build工具,即使用make而不是bjam。后一种方法对我很有效,但现在我想这样做。我通过谷歌搜索和寻找类似的话题解决了很多问题,这帮助我找到了一些窍门并向前迈进。然而,在最后一步,似乎出现了一些问题。我将尝试以一些细节来描述我的所有步骤,希望这篇文章将来可能对其他人有用,也能更好地描述设置 因为我不太清楚python和b

很抱歉这样一个笼统的标题,但我不太确定我到底错过了什么或者我做错了什么。我的目标是在cygwin下使用boost.python构建一个python扩展,并避免使用boost.build工具,即使用make而不是bjam。后一种方法对我很有效,但现在我想这样做。我通过谷歌搜索和寻找类似的话题解决了很多问题,这帮助我找到了一些窍门并向前迈进。然而,在最后一步,似乎出现了一些问题。我将尝试以一些细节来描述我的所有步骤,希望这篇文章将来可能对其他人有用,也能更好地描述设置

因为我不太清楚python和boost的原始安装(来自各种cygwin存储库),所以我决定从头开始在我的主目录中安装它们,所以我的工作如下:

  • 首先安装python。关于这一点,我将略过细节,它或多或少是直截了当的。对于后面的描述,重要的是路径:

    /home/Alexey_2/Soft/python2.6-这是PYTHONPATH,也包含在PATH中

  • 致力于推进:

    a) 将增压源解压缩到

    /home/Alexey_2/Soft/boost_1_50_0   - this is BOOST_ROOT
    
    b) 制作bjam。首先进入目录:

    /home/Alexey_2/Soft/boost_1_50_0/tools/build/v2 
    
    接下来,调用bootstrap.sh,最终在该目录中创建b2和bjam可执行文件。在.bash_概要文件中,将此目录添加到PATH,以便我们可以调用bjam。在这里以及以后每次编辑.bash_概要文件之后,我都会重新启动cygwin进行更改 见效

    c) 还在

    /home/Alexey_2/Soft/boost_1_50_0/tools/build/v2 
    
    directory-edit user-config.jam,让bjam知道要使用哪种python。因此,在我的例子中,我只添加了一行:

    using python : 2.6 : /home/Alexey_2/Soft/python2.6/bin/python2.6 : /home/Alexey_2/Soft/python2.6/include/python2.6 : /home/Alexey_2/Soft/python2.6/bin ;
    
    在lib路径(最后一个条目)中,我放置了/home/Alexey_2/Soft/python2.6/bin,因为它包含libpython2.6.dll

    d) 嗯。现在我们可以制作boostpython库了。转到BOOST_根目录并执行命令

    bjam --with-python toolset=gcc link=shared
    
    这将创建必要的库(cygboost_python.dll和libboost_python.dll.a)并将它们放入

    /home/Alexey_2/Soft/boost_1_50_0/stage/lib 
    
  • 构建python扩展

    这是我的简单测试程序(实际上是示例代码的一部分)

  • 一些评论:

  • 设置了库路径,我根据适当的库进行编译(请参阅更多:)

  • 上面的链接解释了为什么配置user-config.jam很重要——我是在步骤1c中完成的

  • 为了避免boost.python库可能出现的问题(如上面的链接和(尽管对于mingw))我使用

    link=shared 
    
    作为bjam的参数(参见1d)

  • 正如这里所解释的:要编译的库应该列在对象文件之后,这就是为什么我们有:

    ${CPP} ${FLAGS} -shared -o xyz.so xyz.o ${L}
    
    而不是

    ${CPP} ${FLAGS} -shared ${L} -o xyz.so xyz.o
    
  • 下面是我的.bash_配置文件(最终),其中我定义了环境变量:

    # Python
    export PATH=/home/Alexey_2/Soft/python2.6/bin:$PATH
    export PYTHONPATH=/home/Alexey_2/Soft/python2.6
    export LD_LIBRARY_PATH=/home/Alexey_2/Soft/python2.6/lib:/home/Alexey_2/Soft/python2.6/bin:$LD_LIBRARY_PATH
    
    # Boost
    export BOOST_ROOT=/home/Alexey_2/Soft/boost_1_50_0
    export LD_LIBRARY_PATH=/home/Alexey_2/Soft/boost_1_50_0/stage/lib:$LD_LIBRARY_PATH
    export PATH=/home/Alexey_2/Soft/boost_1_50_0/stage/lib:$PATH
    
    # bjam
    export PATH=/home/Alexey_2/Soft/boost_1_50_0/tools/build/v2:$PATH
    
    最后,我想谈谈这个问题。通过上述设置,我能够成功构建python扩展对象文件:

    xyz.so
    
    但是,当我使用简单脚本进行测试时:

    # this is a test.py script
    import xyz
    
    最可怕的是:

    $ python test.py
    Traceback (most recent call last):
      File "test.py", line 1, in <module>
        import xyz
    ImportError: No module named xyz
    
    预期结果(注意python是从该目录到python2.6的符号链接)

    以下是我掌握的另一条有用信息:

    $ ldd xyz.so
        ntdll.dll => /cygdrive/c/Windows/SysWOW64/ntdll.dll (0x76fa0000)
        kernel32.dll => /cygdrive/c/Windows/syswow64/kernel32.dll (0x76430000)
        KERNELBASE.dll => /cygdrive/c/Windows/syswow64/KERNELBASE.dll (0x748e0000)
        cygboost_python.dll => /home/Alexey_2/Soft/boost_1_50_0/stage/lib/cygboost_python.dll (0x70cc0000)
        cygwin1.dll => /usr/bin/cygwin1.dll (0x61000000)
        cyggcc_s-1.dll => /usr/bin/cyggcc_s-1.dll (0x6ff90000)
        cygstdc++-6.dll => /usr/bin/cygstdc++-6.dll (0x6fa90000)
        libpython2.6.dll => /home/Alexey_2/Soft/python2.6/bin/libpython2.6.dll (0x67ec0000)
        ??? => ??? (0x410000)
    
    我在想什么

    ??? => ??? (0x410000)
    
    可能意味着。也许这就是我所缺少的。但那是什么? 非常感谢您的任何意见和建议(不仅仅是关于最后一个问题)

    编辑:

    以下建议(twsansbury)使用-vv选项检查python模块搜索路径:

    python -vv test.py
    
    给予

    生成可以成功加载的xyz.dll:

    python -vv test.py
    
    现在给出:

    Python 2.6.8 (unknown, Mar 21 2013, 17:13:04)
    [GCC 4.5.3] on cygwin
    Type "help", "copyright", "credits" or "license" for more information.
    # trying /home/Alexey_2/Programming/test/xyz.dll
    dlopen("/home/Alexey_2/Programming/test/xyz.dll", 2);
    import xyz # dynamically loaded from /home/Alexey_2/Programming/test/xyz.dll
    

    ImportError通常与Boost.Python无关。相反,它通常表示
    xyz
    不在列表中

    调试此,考虑使用参数运行Python。这将导致python为尝试导入

    xyz
    时检查的每个文件打印一条消息。无论如何,构建过程看起来是正确的,因此问题可能是由于文件扩展名或模块不在搜索路径中

    我不确定Cygwin将如何与Python的运行时加载行为交互。然而:

    • 在上,python扩展有一个
      .pyd
      扩展
    • 在Linux上,python扩展有一个
      .so
      扩展
    此外,请验证
    xyz
    库是否位于以下位置之一:

    • 包含
      test.py
      脚本的目录(或当前目录)
    • 环境变量中列出的目录
    • 依赖于安装的默认目录

    如果
    ldd
    中显示的未解析库导致错误,它通常会显示为
    importorror
    ,并显示一条消息,指示未定义的引用。

    +1问题和研究成果解释得很好。谢谢你,twsansbury!现在一切都很好-我在我原来的帖子中添加了相应的解决方法。
    python -vv test.py
    
    # trying /home/Alexey_2/Programming/test/xyz.dll   
    # trying /home/Alexey_2/Programming/test/xyzmodule.dll
    # trying /home/Alexey_2/Programming/test/xyz.py
    # trying /home/Alexey_2/Programming/test/xyz.pyc
    ...
    # trying /home/Alexey_2/Soft/python2.6/lib/python2.6/site-packages/xyz.dll
    # trying /home/Alexey_2/Soft/python2.6/lib/python2.6/site-packages/xyzmodule.dll
    # trying /home/Alexey_2/Soft/python2.6/lib/python2.6/site-packages/xyz.py
    # trying /home/Alexey_2/Soft/python2.6/lib/python2.6/site-packages/xyz.pyc
    Traceback (most recent call last):
      File "test.py", line 1, in <module>
        import xyz
    ImportError: No module named xyz
    
    all: xyz.dll
    
    xyz.o: xyz.cpp 
            ${CPP} ${FLAGS} ${I} -c xyz.cpp
    
    xyz.dll: xyz.o 
        ${CPP} ${FLAGS} -shared -o xyz.dll xyz.o ${L}
    
    clean:
        rm *.o
        rm xyz.dll
    
    python -vv test.py
    
    Python 2.6.8 (unknown, Mar 21 2013, 17:13:04)
    [GCC 4.5.3] on cygwin
    Type "help", "copyright", "credits" or "license" for more information.
    # trying /home/Alexey_2/Programming/test/xyz.dll
    dlopen("/home/Alexey_2/Programming/test/xyz.dll", 2);
    import xyz # dynamically loaded from /home/Alexey_2/Programming/test/xyz.dll