我设置了Eclipse(实际上是基于Eclipse的Xilinx SDK)和g++4.9.2来编译一个使用独立ASIO的项目,并在Properties->C/C++ Build->Settings->Tool Settings->Other flags中使用了-std=c++11,以便可以使用所有的C++11特性进行编译。
我还在C/C++ General Symbols中设置了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 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 的原因。
ASIO仍然使用 POSIX THREADS 或 Windows 来实现 signal_blocker.hpp 功能,因此仍然需要 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):
#include <thread>
void test() {
for(int i=0;i<100;i++);
}
int main()
{
std::thread thread1(test);
thread1.join();
return 0;
}