在c ++ 11中,std :: atomic可用于在两个线程之间传输非原子数据吗?详细地说,以下4个语义都是由atomic建立的吗?
在原子写入之前执行原子写入语句之前所有语句(在谈论执行时,包括由这些c ++语句生成的所有机器指令)。
在原子读取之后执行原子读取之后的所有语句(在谈论执行时,包括由那些c ++语句生成的所有机器指令)。
写入原子之前的所有其他内存写入都被提交到主内存。
读取原子后读取的所有其他内存读取将再次从主内存读取(这意味着丢弃线程缓存)。
我在这里看到了一个例子:http://bartoszmilewski.com/2008/12/01/c-atomics-and-memory-ordering/
但是,在这个例子中,数据是原子的,所以我的问题是,如果数据是非原子的怎么办?
以下是一些代码,显示了我想要的内容:
常见数据:
std::atomic_bool ready;
char* data; // or data of any other non atomic
写线程:
data = new char[100];
data[0] = 1;
ready.store(true); // use default memory_order(memory_order_seq_cst), witch i think is the most restrict one
阅读帖子:
if(ready.load()) { // use default memory_order(memory_order_seq_cst)
assert(data[0] == 1); // both data(of type char*) and data[0~99](each of type char) are loaded
}
答案 0 :(得分:1)
我认为你必须使用记忆顺序:
data = new char[100];
data[0] = 1;
ready.store_explicit(true, std::memory_order_release);
if(ready.load_explicit(std::memory_order_aqcuire)) {
assert(data[0] == 1); // both data(of type char*) and data[0~99](each of type char) are loaded
}
(我不确定这种语法_explicit
)
实际上你的代码是正确的,但在这种情况下,不需要sec / cst内存顺序,acquire/release
是正确的。
使用release
没有写入可以在原子写入后重新排序,并且acquire
无负载可以在原子加载之前重新排序,因此原子存储之前的所有非原子存储将在原子加载后对所有加载可见。 / p>