我正在阅读“安东尼·威廉姆斯的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;
}