无法在 OS X 上 link 对抗 Boost.Python

Unable to link against Boost.Python on OS X

我正在尝试使用 Boost.Python 构建一个非常简单的示例。我已经用自制软件安装了 boost 和 boost-python 。我正在使用 python 3.4.3 和 boost 1.59。我的 OS 是 El Capitan。

Boost.Python 与 python3 一起安装,如下所示:

brew install boost-python --with-python3

我有 3 个文件:

/* greet.hpp */
#ifndef BOOSTPYTHONHELLOWORLD_GREET_HPP
#define BOOSTPYTHONHELLOWORLD_GREET_HPP

char const* greet();

#endif //BOOSTPYTHONHELLOWORLD_GREET_HPP



/* greet.cpp */    
#include "greet.hpp"

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



/* greet_ext.cpp */
#include "greet.hpp"
#include <boost/python.hpp>

BOOST_PYTHON_MODULE(greet_ext)
{
    using namespace boost::python;
    def("greet", greet);
}

我的 CMakeLists.txt 文件如下所示:

cmake_minimum_required(VERSION 3.3)
project(BoostPythonHelloWorld)
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -std=c++11")
set(PYTHON_LIBRARY "/usr/local/Cellar/python3/3.4.3/Frameworks/Python.framework/Versions/3.4/lib/libpython3.4m.dylib")
set(PYTHON_INCLUDE_DIR "/usr/local/Cellar/python3/3.4.3/Frameworks/Python.framework/Versions/3.4/include/python3.4m")
FIND_PACKAGE(PythonLibs 3.4 REQUIRED)
FIND_PACKAGE(Boost COMPONENTS python)
if(NOT WIN32)
    add_definitions(-DBOOST_ALL_DYN_LINK=1)
    add_definitions(-DBOOST_TEST_DYN_LINK)
endif()
INCLUDE_DIRECTORIES(${Boost_INCLUDE_DIRS} ${PYTHON_INCLUDE_DIRS})
add_library(greet SHARED greet.cpp)
add_library(greet_ext SHARED greet_ext.cpp)
target_link_libraries(greet_ext greet)
target_link_libraries(greet_ext ${PYTHON_LIBRARIES})
target_link_libraries(greet_ext ${Boost_LIBRARIES})
set_target_properties(greet_ext PROPERTIES PREFIX "")

当我 运行 CMake 找到所有正确的 python 库时(因为我手动指定了它们,如您在上面的文件中所见)。

在构建期间我得到以下 link 错误:

Scanning dependencies of target greet
[ 25%] Building CXX object CMakeFiles/greet.dir/greet.cpp.o
[ 50%] Linking CXX shared library libgreet.dylib
[ 50%] Built target greet
Scanning dependencies of target greet_ext
[ 75%] Building CXX object CMakeFiles/greet_ext.dir/greet_ext.cpp.o
[100%] Linking CXX shared library greet_ext.dylib
Undefined symbols for architecture x86_64:
  "boost::python::detail::init_module(PyModuleDef&, void (*)())", referenced from:
      _PyInit_greet_ext in greet_ext.cpp.o
ld: symbol(s) not found for architecture x86_64
clang: error: linker command failed with exit code 1 (use -v to see invocation)
make[2]: *** [greet_ext.dylib] Error 1
make[1]: *** [CMakeFiles/greet_ext.dir/all] Error 2
make: *** [all] Error 2

有人知道为什么会这样吗?

编辑

如果我 link 针对 Python 2.7 它有效,这意味着 boost-python 是针对 python 2.7 而不是 3.4 构建的,即使我指定了 --with-python3 选项..

所以,问题是,如何针对 python 3.4 构建 boost-python?

我目前使用的解决方案是在没有 python 2.7 支持的情况下重新安装 boost-python

我采取的步骤是:

  1. 卸载 boost-python 如果已经安装:brew uninstall boost-python
  2. 安装 boost-python 支持 python3 但不支持 python 2.7:brew install boost-python --with-python3 --without-python.

为了将 boost-python 与 python 3 一起使用,您需要在 CMakeLists.txt 中进行更改:

FIND_PACKAGE(Boost COMPONENTS python)

对此:

FIND_PACKAGE(Boost COMPONENTS python3)

这样你就可以在 python 2 支持下重新编译 boost 并且仍然使用 python 3.

我建议通过 brew 安装 boost-python3

您仍然需要提供次要版本号以及可能的库路径。以下对我有用:

BOOST_LIBRARYDIR=/usr/local/Cellar/boost-python3/1.67.0/lib cmake ..

和CMakeLists.txt(位于..)包含

FIND_PACKAGE(Boost COMPONENTS python36)

(对于 Boost 1.67.0 和 Python 3.6,显然)。