使用 ASIO 和 std::thread 制作 C 11 应用程序时对“pthread_create”的未定义引用错误

新手上路,请多包涵

我设置了 Eclipse(实际上是 Xilinx SDK,但基于 Eclipse)和 g++4.9.2,来编译一个使用独立 ASIO 的项目,我在 Properties -> C/C++ Build -> 中使用了 -std=c++11设置 -> 工具设置 -> 其他标志,以便它可以使用所有 C++11 功能进行编译。

我还在 C/C++ 通用符号中设置 ASIO_HAS_STD_THREAD, ASIO_STANDALONE 等等,并且我希望 ASIO 标头将使用 std::thread 而不是 pthread 。但是,我仍然看到来自 make 的错误:

 undefined reference to pthread_create,
..asio-1.10.6\include\asio\detail\impl\posix_thread.ipp
and posix_tss_ptr.hpp

所以问题是,因为我使用的是 C++11,并且指定了 ASIO_HAS_STD_THREAD 但不是 ASIO_HAS_PTHREADS ,所以 posix_thread.ipp 甚至不应该包括在内(通过 posix_thread. hpp),根据 ASIO 中的 thread.hpp:

 #if !defined(ASIO_HAS_THREADS)
# include "asio/detail/null_thread.hpp"
#elif defined(ASIO_WINDOWS)
# if defined(UNDER_CE)
#  include "asio/detail/wince_thread.hpp"
# else
#  include "asio/detail/win_thread.hpp"
# endif
#elif defined(ASIO_HAS_PTHREADS)
# include "asio/detail/posix_thread.hpp"
#elif defined(ASIO_HAS_STD_THREAD)
# include "asio/detail/std_thread.hpp"
#else
# error Only Windows, POSIX and std::thread are supported!
#endif

嫌疑人1-pthread

与大多数人认为的相反,C++11 不需要 -pthread 并且我尝试在 Eclipse 中编译一个没有 -pthread 的简单项目。但是,如果我错了,您可以纠正我。当我将 -pthread 放在链接器选项中时,它会编译,但是我觉得如果没有必要我不想要 pthread。

嫌疑人 2 - ASIO 生成文件

当我搜索 posix_tss_ptr.hpp 时,我也在 Makefile.am 中找到了。我想知道这是否会影响错误?

那么问题的原因是什么?如果不是以上两个嫌疑人呢?我希望解决方案仍然可以使用纯 C++11 方式,如果我的推理是正确的,不要使用 pthread。

更新

我发现 ASIO_HAS_PTHREADS 不是由我定义的,这就是为什么 ASIO 在某处使用 POSIX 线程,然后链接器需要选项 -pthread。然后我使用#error 指令追溯到 asio/detail/signal_blocker.hpp。它只定义了两个地方,它们在 ASIO config.hpp

 #  if defined(ASIO_HAS_BOOST_CONFIG) && defined(BOOST_HAS_PTHREADS)
#   define ASIO_HAS_PTHREADS 1
#  elif defined(_POSIX_THREADS)
#   define ASIO_HAS_PTHREADS 1

ASIO 仍然在 POSIX THREADS 或 Windows 上回复 signal_blocker.hpp,如下所示。这就是为什么 ASIO 仍然需要 pthread。

 #if !defined(ASIO_HAS_THREADS) || defined(ASIO_WINDOWS) \
  || defined(ASIO_WINDOWS_RUNTIME) \
  || defined(__CYGWIN__) || defined(__SYMBIAN32__)
typedef null_signal_blocker signal_blocker;
#elif defined(ASIO_HAS_PTHREADS)
typedef posix_signal_blocker signal_blocker;
#endif

_PTHREADS 是从 gnu 交叉编译器 (arm-xilinx-linux-gnueabi) 定义的,包括 features.h、posix_opt.h 等文件。我不打算追查真正定义了宏的文件,但 ASIO 是使用 _POSIX_THREADS 的源,因此链接器选项 -pthread 应该在那里。

同样,对于 g++ 4.9.2,非 ASIO C++11 线程不需要 -pthread,但独立 ASIO 需要它。在 g++4.9.2(基于 Eclipse 的 Xilinx SDK)中,以下代码在没有 -pthread 的情况下正确构建:

 #include <thread>
void test() {
    for(int i=0;i<100;i++);
}
int main()
{
    std::thread thread1(test);
    thread1.join();
    return 0;
}

原文由 Splash 发布,翻译遵循 CC BY-SA 4.0 许可协议

阅读 1k
2 个回答

程序是用 C++11 编写的这一事实与是否需要与 pthread 库链接无关。如果它需要 Posix 线程,它需要链接该库。

C++11 提供了 std::thread 类,每个符合标准的编译器的标准库都必须使用目标系统托管的一些本机线程 API 来实现该类的功能。 GCC 使用 pthreads 来实现它,所以你不能用 GCC 构建一个创建 std::thread 对象的程序,除非你将它与 -pthread 链接起来。这个事实与 asio 无关。

之后

我实际上只使用 -std=c++11 使用 std::std 构建了一个没有 pthread 的程序

我认为您对GCC的某些Windows端口默认链接 libpthread 这一事实感到错误和困惑。例如,如果您的示例程序位于 thread.cpp 我可以使用 TDM-GCC 4.9.2 在 Windows 中成功构建它,所以:

 >g++ -std=c++11 -o test_thread thread.cpp

但是如果你以详细模式构建它:

 >g++ -v -std=c++11 -o test_thread thread.cpp

您可以看到很多库选项在幕后传递给链接器,特别是 -lpthread

 >g++ -v -std=c++11 -o test_thread thread.cpp 2>&1 | grep -Po 'pass-through=-lpthread' -
pass-through=-lpthread

在 Linux 上,你不会链接 libpthread 除非你问:

 $ g++ -std=c++11 -o test_thread thread.cpp
/tmp/ccpyEles.o: In function `std::thread::thread<void (&)()>(void (&)())':
thread.cpp:(.text._ZNSt6threadC2IRFvvEJEEEOT_DpOT0_[_ZNSt6threadC5IRFvvEJEEEOT_DpOT0_]+0x7d): undefined reference to `pthread_create'
collect2: error: ld returned 1 exit status

原文由 Mike Kinghan 发布,翻译遵循 CC BY-SA 3.0 许可协议

在执行代码期间使用 g++ ssss(your code name).cpp -std=c++11 -pthread

它会起作用的

原文由 Ramanand Yadav 发布,翻译遵循 CC BY-SA 3.0 许可协议

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题
logo
Stack Overflow 翻译
子站问答
访问
宣传栏