您好我正在阅读实践中的java并发,并且我读到了有趣的声明状态
锁定可以保证可见性和原子性;挥发物 变量只能保证可见性。
任何人都可以解释一下,如果将变量声明为volatile所有其他读取线程获取更新值,那么我为什么要关注语句中的原子性,如:counter = counter + 1
;
提前致谢。
答案 0 :(得分:2)
volatile关键字的作用大致是对该变量的每个单独的读或写操作都是原子的。
然而,值得注意的是,需要多个读/写的操作 - 例如i ++,相当于i = i + 1,它执行一次读取和一次写入 - 不是原子的,因为另一个线程可能在读和写之间写入i。
Atomic类,如AtomicInteger和AtomicReference,以原子方式提供更多种类的操作,特别是包括AtomicInteger的增量。
这就是为什么你需要关心像counter = counter + 1
这样的语句中的原子性请查看此帖子Volatile Vs Atomic
答案 1 :(得分:2)
这是一个自包含的示例可执行应用程序,它演示了volatile本身是不够的。四个线程每个增加一个计数器10,000次,所以你期望计数器最后是40,000。它使用原始int变量和AtomicInt,每次尝试5次。
import java.util.Collections;
import java.util.concurrent.*;
import java.util.concurrent.atomic.AtomicInteger;
class AtomicDemo {
interface Demo extends Callable<Void> {
int getCounter();
}
static class UsePrimitive implements Demo {
private volatile int counter = 0;
public Void call() throws Exception {
for (int i = 1; i <= 10000; ++i) {
++counter;
}
return null;
}
public int getCounter() {
return counter;
}
}
static class UseAtomic implements Demo {
final AtomicInteger counter = new AtomicInteger(0);
public Void call() throws Exception {
for (int i = 1; i <= 10000; ++i) {
counter.incrementAndGet();
System.out.print("");
}
return null;
}
public int getCounter() {
return counter.get();
}
}
public static void main(String[] args) throws Exception {
ExecutorService exec = Executors.newFixedThreadPool(4);
for (int i = 1; i <= 5; ++i) {
Demo demo = new UsePrimitive();
exec.invokeAll(Collections.nCopies(4, demo));
System.out.println("Count to 40000 using primitive, attempt number " + i + ": " + demo.getCounter());
}
for (int i = 1; i <= 5; ++i) {
Demo demo = new UseAtomic();
exec.invokeAll(Collections.nCopies(4, demo));
System.out.println("Count to 40000 using atomic, attempt number " + i + ": " + demo.getCounter());
}
exec.shutdownNow();
}
}
典型输出:
Count to 40000 using primitive, attempt number 1: 39711
Count to 40000 using primitive, attempt number 2: 39686
Count to 40000 using primitive, attempt number 3: 39972
Count to 40000 using primitive, attempt number 4: 39840
Count to 40000 using primitive, attempt number 5: 39865
Count to 40000 using atomic, attempt number 1: 40000
Count to 40000 using atomic, attempt number 2: 40000
Count to 40000 using atomic, attempt number 3: 40000
Count to 40000 using atomic, attempt number 4: 40000
Count to 40000 using atomic, attempt number 5: 40000
你知道,只有使用AtomicInt,你才能获得预期的结果。