1

我回到 c++ 编程,因此,我尝试了很多东西来掌握新的 c++ 标准。我知道我提供的代码非常糟糕。我想我知道如何解决它。

代码:

#include <iostream>
#include <thread>

using namespace std;

void apple (string const& x)
{
                cout << "aaa" << endl;
}

void orange()
{
                //string s = "hello";
                thread t(apple,"hello");
                t.detach();
}

int main() {
                orange();
                cout << "!!!Hello World!!!" << endl; // prints !!!Hello World!!!
                return 0;
}

一种输出:

!!!Hello World!!!
aaa
aaa

从代码中,我们可以看到“aaa”应该打印一次。我的问题是为什么输出上有两个“aaa”。是什么导致了这个问题?

非常感谢

编辑:添加系统信息:

Using built-in specs.
COLLECT_GCC=/usr/bin/g++
COLLECT_LTO_WRAPPER=/usr/libexec/gcc/x86_64-redhat-linux/4.6.3/lto-wrapper
Target: x86_64-redhat-linux
Configured with: ../configure --prefix=/usr --mandir=/usr/share/man --infodir=/usr/share/info --with-bugurl=http://bugzilla.redhat.com/bugzilla --enable-bootstrap --enable-shared --enable-threads=posix --enable-checking=release --with-system-zlib --enable-__cxa_atexit --disable-libunwind-exceptions --enable-gnu-unique-object --enable-linker-build-id --enable-languages=c,c++,objc,obj-c++,java,fortran,ada,go,lto --enable-plugin --enable-java-awt=gtk --disable-dssi --with-java-home=/usr/lib/jvm/java-1.5.0-gcj-1.5.0.0/jre --enable-libgcj-multifile --enable-java-maintainer-mode --with-ecj-jar=/usr/share/java/eclipse-ecj.jar --disable-libjava-multilib --with-ppl --with-cloog --with-tune=generic --with-arch_32=i686 --build=x86_64-redhat-linux
Thread model: posix
gcc version 4.6.3 20120306 (Red Hat 4.6.3-2) (GCC) 

其他一些输出:

A型:

!!!Hello World!!!
aaaaaa

B型:

!!!Hello World!!!

C型:

!!!Hello World!!!
aaaaaa

D型:

!!!Hello World!!!
aaa
<empty line>

编辑:根据@mark的回答。这是一个未定义的行为。
我添加了一个“睡眠(1);” 在主要返回之前。它只给我一种类型的输出。现在我很困惑,如果它是未定义的,为什么我没有看到其他类型的输出?

输出:

!!!Hello World!!!
aaa
4

1 回答 1

3

std::cout除非您使用真正的 c++11 编译器,否则它不是线程安全的。在没有保护的情况下,您不能有 2 个线程写入 cout。如果你这样做了,你会得到未定义的行为——就像你看到的那样!

请参阅此讨论:cout 是否同步/线程安全?

c++11 似乎直到gcc 4.7才得到完全支持,而你有 gcc 4.6

于 2013-09-12T20:36:57.703 回答