10

我正在尝试构建我的第一个 Boost.Python 示例。

#include <iostream>
#include <boost/python.hpp>

using namespace boost::python;


class Hello {

public:
    std::string greet() {
        std::cout << "Hello World" << std::endl;
    }
};


BOOST_PYTHON_MODULE(hello)
{
    class_<Hello>("Hello")
        .def("greet", &Hello::greet);
}

int main() {
    std::cout << "Boost.Python Test" << std::endl;
    Hello hello;
    hello.greet();
    return 0;
}

编辑:正如@cdhowie 所指出的,缺少 Python 开发标头。我已经找到并包含了所需的头文件。现在链接器抱怨:

   10:43:58 **** Build of configuration BoostPythonTest-DPar for project BoostPythonTest 

****
make all 
Building file: ../src/BoostPythonTest.cpp
Invoking: GCC C++ Compiler
/usr/local/bin/g++-4.7 -I/usr/local/Cellar/python3/3.3.0/Frameworks/Python.framework/Versions/3.3/include/python3.3m -I/usr/include -I/usr/local/Cellar/gcc/4.7.2/gcc/include/c++/4.7.2 -O0 -g3 -p -pg -Wall -c -fmessage-length=0 -std=c++11 -MMD -MP -MF"src/BoostPythonTest.d" -MT"src/BoostPythonTest.d" -o "src/BoostPythonTest.o" "../src/BoostPythonTest.cpp"
Finished building: ../src/BoostPythonTest.cpp

Building target: libBoostPythonTest-DPar.dylib
Invoking: MacOS X C++ Linker
/usr/local/bin/g++-4.7 -L/usr/local/Cellar/python3/3.3.0/Frameworks/Python.framework/Versions/3.3/lib/python3.3/config-3.3m -L/usr/local/Cellar/python3/3.3.0/Frameworks/Python.framework/Versions/3.3/lib -L/usr/local/Cellar/boost/1.51.0/lib -std=c++11 -Xlinker -ldl -framework CoreFoundation -lpython3.3m -dynamiclib -o "libBoostPythonTest-DPar.dylib"  ./src/BoostPythonTest.o   -lpython3.3m -lboost_python-mt -lpython3.3
Undefined symbols for architecture x86_64:
  "boost::python::detail::init_module(PyModuleDef&, void (*)())", referenced from:
      _PyInit_hello in BoostPythonTest.o
ld: symbol(s) not found for architecture x86_64
collect2: error: ld returned 1 exit status
make: *** [libBoostPythonTest-DPar.dylib] Error 1

我已经链接到-lpython3.3m -lboost_python-mt -lpython3.3- 还缺少什么?

编辑:我想我已经链接到python3.3-config列出的所有内容。由于缺少符号,链接仍然不起作用。

4

5 回答 5

13

当发生这种特定的链接器错误时,通常是应用程序针对一个 Python 版本(例如 Python 3.x 头文件)构建的结果,而该boost_python库是针对不同版本(例如 2.x)构建的。

boost/python/module_init.hpp中,该init_module函数在针对 Python 3.x 构建时具有以下签名:

PyObject* boost::python::detail::init_module(PyModuleDef&, void(*)());

以及针对 Python 2.x 构建时的以下签名:

PyObject* boost::python::detail::init_module(char const* name, void(*)());

从实现中可以看出,Boost.Python 库中只有一个函数。因此,鉴于正在链接 Boost.Python 库,而链接器只是抱怨无法解析 3.xinit_module函数,那么 Boost.Python 库很可能是针对 Python 2.x 构建的版本,而应用程序代码是针对 Python 3.x 头文件构建的。您可以通过转储 Boost.Python 库的符号并检查init_module签名来验证这一点。

要解决此问题,请使用构建 Boost.Python 的同一版本的 Python 构建应用程序。在这种情况下,要么:

  • 使用 Python 2.x 头文件构建应用程序并链接到 Python 2.x 库。
  • 针对 Python 3.x 构建 Boost.Python。 文档描述了构建 Boost 的步骤,文档详细介绍了 Boost.Python。可能有必要使用参数显式地提供 Boost.Python 将在此bootstrap过程中构建的 Python 可执行文件--with-python
于 2013-04-16T15:19:16.177 回答
6

您缺少 Python 开发标头。你的 Linux 发行版应该有一个包。(例如,python-dev在 Debian 或 Ubuntu 上。)

于 2013-04-10T15:05:59.617 回答
2

您正在构建一个共享库,因为这就是二进制 Python 模块。为此,您需要 -shared 或 -dynamic (查看文档),并且您不应该有 main() 函数。

此外,如果这没有帮助并且您仍然有链接器错误,请使用“objdump -T --demangle /path/to/lib”来找出库包含哪些符号以及它是否具有您需要的符号。还要检查“ldd”的输出,它列出了依赖的共享对象。这应该会提示您链接哪个或哪些库。

在我的系统上,我还有一个名为“python-config”和“python2.7-config”的程序。检查你是否有类似的东西,因为这个脚本知道至少要为 python 链接哪些库。一个类似的工具是 pkg-config,它更通用,也可能为 boost 提供信息。

于 2013-04-10T19:45:12.213 回答
2

不确定这是否是这样做的方法,但似乎 PY_VERSION_HEX 设置错误。如果你会发生什么

#define PY_VERSION_HEX 0x03300000

在您的 BoostPythonTest.cpp 中包含 boost.python 标头之前?

于 2013-04-22T20:47:43.760 回答
1

您还必须与 Boost.System 以及链接器提到的任何其他库进行链接。此外,尝试其他编译器,例如XCode 附带的Clang( https://svn.boost.org/trac/boost/ticket/7536 )( XCode 4.5 中的最新 Clang 位置)。另外,尝试发布您的makefile。另一种尝试是使用 bjam/b2 而不是 make。

于 2013-04-17T18:30:15.237 回答