gpt4 book ai didi

c++ - undefined reference `pthread_create' 使用 ASIO 和 std::thread 制作 C++11 应用程序时出错

转载 作者:塔克拉玛干 更新时间:2023-11-02 23:04:51 24 4
gpt4 key购买 nike

我设置了 Eclipse(实际上是 Xilinx SDK,但基于 Eclipse)和 g++4.9.2 来编译一个使用独立 ASIO 的项目,我在属性 -> C/C++ 中使用了 -std=c++11 Build -> Settings -> Tool Settings -> Other flags 以便它可以使用所有 C++11 特性进行编译。

我还在C/C++ General Symbols中设置了ASIO_HAS_STD_THREAD, ASIO_STANDALONE等等,我希望ASIO头文件会使用std::thread而不是线程。但是,我仍然看到来自 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 应该根据 ASIO 中的 thread.hpp,甚至不包括(通过 posix_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 makefile

当我搜索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;
}

最佳答案

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

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

稍后

I actually built a program without pthread using std::std just using -std=c++11

我认为你对 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

关于c++ - undefined reference `pthread_create' 使用 ASIO 和 std::thread 制作 C++11 应用程序时出错,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34933042/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com