线程原子类:memory_order_seq_cst VS memory_order_relaxed

时间:2017-09-05 14:18:27

标签: c++ atomic

我正在阅读“安东尼·威廉姆斯的C ++并发行动”一书来研究并发性。这是一本很棒的书。我觉得很难理解原子类中的内存模型之间的区别。例如,memory_order_seq_cst与memory_order_relaxed相比,我希望有人可以更好地解释它们。以下是本书清单5.5中的代码。我添加了memory_order_seq_cst的代码来测试与memory_order_relaxed的差异。但很难看出与输出的区别。也不是书中的解释。

谢谢, L

#include <iostream>
#include <thread>
#include <atomic>

std:: atomic<int> x(0),y(0),z(0);
std:: atomic<bool> go(false);

unsigned const loop_count=10;

struct read_values
{
    int x,y,z;
};
read_values values1[loop_count];
read_values values2[loop_count];
read_values values3[loop_count];
read_values values4[loop_count];
read_values values5[loop_count];

void increment(std::atomic<int>* var_to_inc,read_values* values)
{
    while(!go)
        std::this_thread::yield();  // !!!!!
    for (unsigned i=0;i<loop_count;++i)
    {
//        values[i].x=x.load(std::memory_order_relaxed);
//        values[i].y=y.load(std::memory_order_relaxed);
//        values[i].z=z.load(std::memory_order_relaxed);
//        var_to_inc->store(i+1,std::memory_order_relaxed);
        values[i].x=x.load(std::memory_order_seq_cst);
        values[i].y=y.load(std::memory_order_seq_cst);
        values[i].z=z.load(std::memory_order_seq_cst);
        var_to_inc->store(i+1,std::memory_order_seq_cst);
        std::this_thread::yield(); // !!!!!
    }
}
void read_vals(read_values* values)
{
    while (!go)
        std:: this_thread::yield();
    for (unsigned i=0;i<loop_count;++i)
    {
//        values[i].x=x.load(std::memory_order_relaxed);
//        values[i].y=y.load(std::memory_order_relaxed);
//        values[i].z=z.load(std::memory_order_relaxed);
        values[i].x=x.load(std::memory_order_seq_cst);
        values[i].y=y.load(std::memory_order_seq_cst);
        values[i].z=z.load(std::memory_order_seq_cst);
        std::this_thread::yield(); // !!!!!
    }
}
void print(read_values* v)
{
    for (unsigned i=0;i<loop_count;++i)
    {
        if (i)
            std::cout<<",";
        std::cout<<"("<<v[i].x<<","<<v[i].y<<","<<v[i].z<<")";
    }
    std::cout<<std::endl;
}
int main(int argc, const char * argv[])
{
    std::thread t1(increment, &x,values1);
    std::thread t2(increment, &y,values2);
    std::thread t3(increment, &z,values3);
    std::thread t4(read_vals,values4);
    std::thread t5(read_vals,values5);
    go=true;
    t5.join();
    t4.join();
    t3.join();
    t2.join();
    t1.join();
    print(values1);
    print(values2);
    print(values3);
    print(values4);
    print(values5);
    return 0;
}

0 个答案:

没有答案