使用g++编译多线程代码

95
我有史上最简单的代码:

我有史上最简单的代码:

#include <iostream>
#include <thread>

void worker()
{
    std::cout << "another thread";
}

int main()
{
    std::thread t(worker);
    std::cout << "main thread" << std::endl;
    t.join();
    return 0;
}

虽然我仍然无法使用g++编译它并运行。

更多细节:

$ g++ --version
g++ (Ubuntu/Linaro 4.8.1-10ubuntu8) 4.8.1
Copyright (C) 2013 Free Software Foundation, Inc.
This is free software; see the source for copying conditions.  There is NO
warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.

编译命令:

$ g++ main.cpp -o main.out -pthread -std=c++11

运行中:

$ ./main.out 
terminate called after throwing an instance of 'std::system_error'
  what():  Enable multithreading to use std::thread: Operation not permitted
Aborted (core dumped)

我现在陷入了困境。在互联网上的所有相关线程中,都建议添加-pthread,而我已经添加了。

我做错了什么?

附注:这是一个全新的Ubuntu 13.10安装。只安装了g++软件包和一些小东西,如chromium等。

附注2:

$ ldd ./a.out 
linux-vdso.so.1 => (0x00007fff29fc1000) 
libstdc++.so.6 => /usr/lib/x86_64-linux-gnu/libstdc++.so.6 (0x00007fb85397d000) 
libgcc_s.so.1 => /lib/x86_64-linux-gnu/libgcc_s.so.1 (0x00007fb853767000) 
libc.so.6 => /lib/x86_64-linux-gnu/libc.so.6 (0x00007fb85339e000) 
libm.so.6 => /lib/x86_64-linux-gnu/libm.so.6 (0x00007fb85309a000) 
/lib64/ld-linux-x86-64.so.2 (0x00007fb853c96000)

附注:使用 clang++ (v3.2)进行编译和运行没有问题。

各位,这并不是在Linux下使用GCC和std::thread的正确链接选项是什么?的重复问题。

再附注:

$ dpkg --get-selections | grep 'libc.*dev'
libc-dev-bin                    install
libc6-dev:amd64                 install
libclang-common-dev             install
linux-libc-dev:amd64                install

它是同一目录下指向 libpthread-2.17.so 的符号链接。 - zerkms
是的,那是一个正确的链接。这是我的目录:lrwxrwxrwx. 1 root root 18 Sep 12 20:52 libpthread.so.0 -> libpthread-2.16.so - Jason Enochs
@Jason Enochs:如果我没有使用错误的gcc版本,这对我很有效 - 我同意。 - zerkms
是的,你所有的库都比我的新一点: gcc版本4.7.2 20121109(Red Hat 4.7.2-8)(GCC) - Jason Enochs
@Jason Enochs:呵呵。对于我自己来说,“这是一个 bug” 的答案已经让我满意了,明天我可能会尝试在旧版的 Ubuntu 上运行它,以确保完全正确。 - zerkms
4个回答

75

答案由一个友好的 SO C++ chat 成员提供。

看起来这种行为是由gcc中的bug引起的。

该错误讨论的最后一条评论提供的解决方法确实可行,可以解决此问题:

-Wl,--no-as-needed

9
没有你和像你这样的人,我该怎么办?你们帮我省去了很多烦恼。谢谢 :) 这对我来说够晦涩难懂的了。 - scorpiodawg
@scorpiodawg:你的评论让我今晚过得很愉快,谢谢 :-) - zerkms
2
我不得不添加所有的-pthread -lpthread -Wl,--no-as-needed标志才能使它工作。 - The Paramagnetic Croissant

39

在我的情况下,添加-lpthread解决了相同的问题:

 g++ -std=c++11 foo.cpp -lpthread -o foo

1
是的,这就是答案,而且完全合理。“-Wl,--no-as-needed”听起来很奇怪,对我也没有用。 - thesaint
3
你肯定需要-lpthread。取决于编译器版本,你可能需要-pthread -Wl,--no-as-needed,对于gcc-4.8.2来说是必须的。 - cdunn2001

15

我有一个稍微先进一些的版本(4.8.4而不是4.8.1),并且我测试了上面的所有三个答案。实际上:

-pthread 单独使用可以工作:

g++ -std=c++11 -o main -pthread main.cpp

-Wl,--no-as-needed 单独无法工作

-lpthread 单独无法工作

-Wl,--no-as-needed-lpthread 一起使用 可以工作:

g++ -std=c++11 -o main -Wl,--no-as-needed main.cpp -lpthread

我的版本为“g++ (Ubuntu 4.8.4-2ubuntu1~14.04.1) 4.8.4”。


好的,这些解决方案是针对不同版本提供的,我可以保证在我检查时,已经验证过的答案是有效的。无论如何,感谢您提供的实际信息。 - zerkms

10

已经有关于Qt Creator的答案了:

LIBS += -pthread
QMAKE_CXXFLAGS += -pthread
QMAKE_CXXFLAGS += -std=c++11

对于控制台g++: 这里

g++ -c main.cpp -pthread -std=c++11         // generate target object file
g++ main.o -o main.out -pthread -std=c++11  // link to target binary

1
谢谢,您的回答是唯一一个清楚地表明必须将-pthread传递给编译器和链接器的答案。 - denim

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接