c++ - 无法链接到 boost 库

标签 c++ boost g++ ubuntu-14.04 boost-coroutine

我无法让我的 g++ 在我的机器上编译和链接一个最小的 boost 协程程序(我想还有其他组件)。

// example.cpp

#include <boost/coroutine/all.hpp>
#include <iostream>
using namespace boost::coroutines;

void mycorofunc(coroutine<void>::push_type &sink){
  std::cout << "1";
  sink();
  std::cout << " 3";
}

int main() {
  coroutine<void>::pull_type source{mycorofunc};
  std::cout << " 2";
  source();
  std::cout << " 4!" << std::endl;
}

我使用 apt-get 安装了 boost (libboost-all-dev),但都没有

g++ example.cpp -lboost_coroutine -lboost_system

也不

g++ example.cpp -I/usr/include -L/usr/lib/x86_64-linux-gnu -lboost_coroutine -lboost_system

有效。 g++的输出如下:

In file included from /usr/include/boost/coroutine/v1/coroutine.hpp:19:0,
                 from /usr/include/boost/coroutine/coroutine.hpp:13,
                 from /usr/include/boost/coroutine/all.hpp:11,
                 from example.cpp:3:
/usr/include/boost/type_traits/function_traits.hpp: In instantiation of ‘struct boost::function_traits<void>’:
example.cpp:7:31:   required from here
/usr/include/boost/type_traits/function_traits.hpp:168:8: error: invalid use of incomplete type ‘struct boost::detail::function_traits_helper<void*>’
 struct function_traits : 
        ^
/usr/include/boost/type_traits/function_traits.hpp:21:36: error: declaration of ‘struct boost::detail::function_traits_helper<void*>’
 template<typename Function> struct function_traits_helper;
                                    ^
example.cpp:7:31: error: ‘arity’ is not a member of ‘boost::function_traits<void>’
 void mycorofunc(coroutine<void>::push_type &sink){
                               ^
example.cpp:7:31: error: template argument 2 is invalid
example.cpp:7:44: error: expected ‘,’ or ‘...’ before ‘&’ token
 void mycorofunc(coroutine<void>::push_type &sink){
                                            ^
example.cpp: In function ‘void mycorofunc(int)’:
example.cpp:9:8: error: ‘sink’ was not declared in this scope
   sink();
        ^
example.cpp: In function ‘int main()’:
example.cpp:14:17: error: ‘arity’ is not a member of ‘boost::function_traits<void>’
   coroutine<void>::pull_type source{mycorofunc};
                 ^
example.cpp:14:17: error: template argument 2 is invalid
example.cpp:14:30: error: expected initializer before ‘source’
   coroutine<void>::pull_type source{mycorofunc};
                              ^
example.cpp:16:10: error: ‘source’ was not declared in this scope
   source();

关于我的环境的更多信息:

$$ locate coroutine/all.hpp
/usr/include/boost/coroutine/all.hpp
/usr/local/boost_1_58_0/boost/coroutine/all.hpp

$$ locate boost_system
/usr/lib/x86_64-linux-gnu/libboost_system.a
/usr/lib/x86_64-linux-gnu/libboost_system.so
/usr/lib/x86_64-linux-gnu/libboost_system.so.1.54.0

$$ locate boost_coroutine
/usr/lib/x86_64-linux-gnu/libboost_coroutine.a

$$ g++ -E -x c++ - -v < /dev/null  # printing g++ include path 
Using built-in specs.
COLLECT_GCC=g++
Target: x86_64-linux-gnu
Configured with: ../src/configure -v --with-pkgversion='Ubuntu 4.8.2-19ubuntu1' --with-bugurl=file:///usr/share/doc/gcc-4.8/README.Bugs --enable-languages=c,c++,java,go,d,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-4.8 --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --with-gxx-include-dir=/usr/include/c++/4.8 --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --enable-gnu-unique-object --disable-libmudflap --enable-plugin --with-system-zlib --disable-browser-plugin --enable-java-awt=gtk --enable-gtk-cairo --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-4.8-amd64/jre --enable-java-home --with-jvm-root-dir=/usr/lib/jvm/java-1.5.0-gcj-4.8-amd64 --with-jvm-jar-dir=/usr/lib/jvm-exports/java-1.5.0-gcj-4.8-amd64 --with-arch-directory=amd64 --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --enable-objc-gc --enable-multiarch --disable-werror --with-arch-32=i686 --with-abi=m64 --with-multilib-list=m32,m64,mx32 --with-tune=generic --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=x86_64-linux-gnu
Thread model: posix
gcc version 4.8.2 (Ubuntu 4.8.2-19ubuntu1) 
COLLECT_GCC_OPTIONS='-E' '-v' '-shared-libgcc' '-mtune=generic' '-march=x86-64'
 /usr/lib/gcc/x86_64-linux-gnu/4.8/cc1plus -E -quiet -v -imultiarch x86_64-linux-gnu -D_GNU_SOURCE - -mtune=generic -march=x86-64 -fstack-protector -Wformat -Wformat-security
ignoring duplicate directory "/usr/include/x86_64-linux-gnu/c++/4.8"
ignoring nonexistent directory "/usr/local/include/x86_64-linux-gnu"
ignoring nonexistent directory "/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../../x86_64-linux-gnu/include"
#include "..." search starts here:
#include <...> search starts here:
 /usr/include/c++/4.8
 /usr/include/x86_64-linux-gnu/c++/4.8
 /usr/include/c++/4.8/backward
 /usr/lib/gcc/x86_64-linux-gnu/4.8/include
 /usr/local/include
 /usr/lib/gcc/x86_64-linux-gnu/4.8/include-fixed
 /usr/include/x86_64-linux-gnu
 /usr/include
End of search list.
# 1 "<stdin>"
# 1 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 1 "<command-line>" 2
# 1 "<stdin>"
COMPILER_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/
LIBRARY_PATH=/usr/lib/gcc/x86_64-linux-gnu/4.8/:/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../x86_64-linux-gnu/:/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../../lib/:/lib/x86_64-linux-gnu/:/lib/../lib/:/usr/lib/x86_64-linux-gnu/:/usr/lib/../lib/:/usr/lib/gcc/x86_64-linux-gnu/4.8/../../../:/lib/:/usr/lib/
COLLECT_GCC_OPTIONS='-E' '-v' '-shared-libgcc' '-mtune=generic' '-march=x86-64'

编译器:已尝试 g++ (v. 4.8.2) 和 g++-5

发行版:Ubuntu 14.04.1 LTS

最佳答案

经过几天的调试,这是我找到的解决方案。有可能即使是以下说明的一个子集也足以解决该问题。

没有太多细节的快速说明

执行以下指令:

  1. 安装 g++-6(任何高于 5.2 且支持 c++14 的版本都将 很可能也能正常工作,尽管我还没有测试过它们)。
  2. 安装 boost 1.58 或更新版本(支持 coroutine2)。截至日期 这篇文章,Ubuntu PPA 中可用的最新版本是 1.54。因此,您需要手动安装boost。不懂的可以关注this link .
  3. 使用 -std=c++14 标志运行您的程序。此外,如果您像我一样使用协程,请确保链接到 libboost_context。我注意到对于协程 2,不需要 libboost_system,可能是因为它被 libboost_context 自动引入,但我不确定关于原因。
  4. 确保链接到新安装的 boost 库,而不是旧库。参见 this SO question看看可能出了什么问题。就我而言,我必须从我的机器上删除所有以前安装的 boost 。

为了将这些点放在一起,这是我用来编译和链接我的代码的命令(假设我在 /usr/local/ 中安装了 boost):

g++-6 -std=c++14 example.cpp -I/usr/local/include -L/usr/local/lib -lboost_coroutine -lboost_context -lboost_system

更多详情

显然,boost 1.54 中的 boost.coroutine 库是使用 boost.context [ source 中已弃用的类似 C 的 fcontext-API 实现的。 ].我注意到我的机器在编译这些部分时遇到了问题。不幸的是,目前 Ubuntu PPA 不提供任何比 1.54 更新的版本。 Boost.coroutine2 是推荐使用的库,可以在 1.59 或更高版本中找到。因此,您需要手动从其网站下载合适版本的boost并安装。

这样做之后,我的代码成功编译,但没有链接:

... undefined reference to `ontop_fcontext'  `boost::context::basic_fixedsize_stack<boost::context::stack_traits>::deallocate(boost::context::stack_context&)':
example.cpp:(...): undefined reference to `boost::context::stack_traits::minimum_size()'
example.cpp:(...): undefined reference to `boost::context::stack_traits::is_unbounded()'
example.cpp:(...): undefined reference to `boost::context::stack_traits::maximum_size()'

令人惊讶的是,将 -lboost_context 标志添加到编译命令并没有消除此链接错误,这表明 libboost_context 未正确链接。

然后我遇到了 a question在 boost 论坛中,原始发帖人的问题与我遇到的问题有些相似。简而言之,有许多 C 宏会阻止正确包含 execution_context,从而导致 libboost_context 出现问题。了解您的编译器是否启用了其中一些宏的代码如下(我不确定是否所有这些宏都相关):

#include <boost/config.hpp> 

#if defined(BOOST_NO_CXX11_AUTO_DECLARATIONS) || \
     defined(BOOST_NO_CXX11_CONSTEXPR) || \
     defined(BOOST_NO_CXX11_DEFAULTED_FUNCTIONS) || \
     defined(BOOST_NO_CXX11_FINAL) || \
     defined(BOOST_NO_CXX11_HDR_TUPLE) || \
     defined(BOOST_NO_CXX11_NOEXCEPT) || \
     defined(BOOST_NO_CXX11_NULLPTR) || \
     defined(BOOST_NO_CXX11_RVALUE_REFERENCES) || \
     defined(BOOST_NO_CXX11_TEMPLATE_ALIASES) || \
     defined(BOOST_NO_CXX11_UNIFIED_INITIALISATION_SYNTAX) || \
     defined(BOOST_NO_CXX11_VARIADIC_TEMPLATES) || \
     defined(BOOST_NO_HDR_ATOMIC) || \
     defined(BOOST_NO_HDR_TUPLE)
#error "execution_context is prevented to be included in this compiler"; 
#endif 

解决方案很简单:安装g++-6(或者我猜想 5.2 或更高版本),然后使用 编译代码std=-c++14.

关于c++ - 无法链接到 boost 库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41413791/

相关文章:

c++ - C++中的新行错误

c++ - 无法使用 Visual Studio 2015 Pro 构建 boost 1.60

c++ - 是否有任何 g++ 选项可以转储类布局和 vtables?

directory - 如何让 g++ 搜索特定目录中的头文件?

c++ - 类内外无作用域的枚举重定义

C++ 单头文件结构

c++ - 高效的 boost 分配使用

dll - g++ 无法识别 dll 的文件格式

c++ - 我应该使用哪些编译标志来避免运行时错误

c++ - 用于检索集合中第 k 个最小/最大项目的数据结构(STL 或 Boost)?