顺序一致性和原子性的区别是什么?
What is the difference between sequential consistency and atomicity?
我读到java volatile是顺序一致的,但不是原子的。对于原子性java提供了不同的库。
谁能用简单的英语解释一下两者的区别?(我相信问题范围包括C/c++,因此添加这些语言标签可以获得更多的受众)
想象这两个变量在一个类中:
int i = 0;
volatile int v = 0;
和这两个方法
void write() {
i = 5;
v = 2;
}
void read() {
if (v == 2) { System.out.println(i); }
}
volatile语义保证read
要么打印5,要么不打印(当然假设没有其他方法修改字段)。如果v
不是易失性的,read
也可以打印0,因为i = 5
和v = 2
可能已经重新排序了。我猜这就是你所说的顺序一致性,它有更广泛的含义。
另一方面,volatile不保证原子性。所以如果两个线程同时调用这个方法(v是同一个volatile int
):
void increment() {
v++;
}
你不能保证v会加2。这是因为v++
实际上是三个语句:
load v;
increment v;
store v;
假设您有以下两个变量:
public int a = 0;
public volatile int b = 0;
假设一个线程执行
a = 1;
b = 2;
如果另一个线程读取这些值并看到b == 2,那么它肯定也会看到a == 1。
但是读线程可以看到a == 1
和b == 0
,因为这两个写操作不是原子操作的一部分,所以读线程可能在第一个线程给b
赋值之前看到a
的变化。
要使这两个写操作原子化,需要同步对这两个变量的访问:
synchronized (lock) {
a = 1;
b = 2;
}
...
synchronized (lock) {
System.out.println("a = " + a + "; b = " + b);
}
在这种情况下,读线程将看到a == 0
和b == 0
,或者a == 1
和b == 2
,但永远不会看到中间状态
相关文章:
- 为不同配置设置MSVC_RUNTIME_LIBRARY的正确方法是什么
- C++避免重复声明的语法是什么
- 在C++中,将大的无符号浮点数四舍五入为整数的最佳方法是什么
- 实现无开销push_back的最佳方法是什么
- C++从另一个类访问公共静态向量的正确方法是什么
- "throw expression code" 1e7 >返回 d 是什么?投掷标准::overflow_error( "too big" ) : d;意味 着?
- C++中名称篡改的目的是什么
- 在 c++ 中拥有一组结构的正确方法是什么?
- 这个指针和内存代码打印是什么?我不知道是打印垃圾还是如何打印我需要的值
- 是什么阻止DOMTimerCoordinator::NextID进入无休止的循环
- 派生类销毁的最佳实践是什么
- 这个语法std::class<>{}(arg1, arg2) 在C++中是什么意思?
- 通过JNI传递数据数组的最快方法是什么
- "using namespace std;"在C++的作用是什么?
- C++和Python之间平等含义的区别的根源是什么?
- 尾随空格是什么意思,它和空白有什么区别?
- 调试构建和发布构建、区别和用途是什么意思
- "Thing thing;"和"Thing thing = Thing();"有什么区别,什么时候应该使用一个而不是另一个?
- char和CString的区别和关系是什么?
- 那么,是什么区别了template(c.end(),_1)和template_back(_1)呢