- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
Java 通过其原子类公开 CAS 操作,例如
boolean compareAndSet(expected,update)
JavaDocs指定 compareAndSet 操作的内存效果如下:
compareAndSet and all other read-and-update operations such as getAndIncrement have the memory effects of both reading and writing volatile variables.
这绝对适用于成功的 compareAndSet
调用。但是,如果 compareAndSet 返回 false
,内存效应是否也成立?
我会说不成功的 compareAndSet
对应于 volatile 读取(因为在这种情况下必须访问原子实例的当前值),但我不明白为什么 CAS 应该执行不成功情况下的特殊内存屏障指令。
问题实际上是,不成功的 CAS 是否也建立了 happens-before 关系。考虑以下程序:
public class Atomics {
private static AtomicInteger ai = new AtomicInteger(5);
private static int x = 0;
public static void main(String[] args) {
new Thread(() -> {
while (x == 0) {
ai.compareAndSet(0, 0); // returns false
}
}, "T1").start();
new Thread(() -> {
x = 1;
ai.compareAndSet(0, 0); // returns false
}, "T2").start();
}
}
线程 T2(和程序)一定会终止吗?
最佳答案
使用 volatile
读取和写入建立happens-before 关系的问题在于,这种关系只存在于写入和后续 阅读。如果一个线程 T1 写入一个共享的 volatile
变量,而另一个线程 T2 从同一个变量读取,如果 T2 之前读取该变量,则不存在happens-before 关系T1 写信给它。如果决定 T1 是否在 T2 读取之前写入的只是线程调度,那么我们没有任何保证。
在没有额外同步的情况下处理它的一种实用方法是评估实际值,T2 已读取。如果此值表明 T1 已写入新值,则我们有一个有效的happens-before 关系。这就是它在使用 volatile boolean fooIsInitialized
标志或 volatile int currentPhase
计数器时的工作方式。很明显,如果写入的值与旧值相同,或者如果新值从未实际写入,这将无法工作。
您的示例程序的问题在于它推测线程调度。它假设 T2 最终执行了 cas 操作,并且在 T1 中将有一个后续迭代,其中下一个 cas 将创建一个先发生关系。但这并不能保证。这可能不是直觉上可以理解的,但如果没有同步,T1 的所有迭代都可能在 T2 的操作之前发生,即使循环是无限的。它甚至是一种有效的线程调度行为,让 T1 永远消耗 100% 的 CPU 时间,然后再将 CPU 时间分配给 T2,因为无法保证在相同优先级的线程之间进行抢占式线程切换。
但是,即使底层系统确实将 CPU 时间分配给最终会执行操作的 T2,JVM 也不需要让 T1 明白这一点,因为 T1 无法观察到 T2 曾经运行过。在现实生活中不太可能发现这一点,但答案仍然是没有保证。当存在一系列操作使 T1 可以观察到 T2 运行(即更改其状态)时,情况会发生变化,但当然,该操作链会使 cas 过时。
关于java - compareAndSet 不成功操作的内存效应,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29846319/
Java 通过其原子类公开 CAS 操作,例如 boolean compareAndSet(expected,update) JavaDocs指定 compareAndSet 操作的内存效果如下: c
假设您有以下类(class) public class AccessStatistics { private final int noPages, noErrors; public Acces
正在学习并发编程类(class)。 举个例子 final class Counter { private AtomicInteger value; public lon
我想知道原子类中 set() 和 compareAndSet() 的区别。 set() 方法是否也保证了原子过程?例如这段代码: public class sampleAtomic{ priv
今天我在采访中被问到下一个问题:“如果您在处理器不支持 CAS 操作的机器上调用 AtomicLong 的 compareAndSet 方法,会发生什么情况”。 能否请您帮我解决这个问题,并在可能的情
我正在实现一个简单的缓存,并将缓存存储为 AtomicReference。 private AtomicReference> cacheData; 应该从数据库表中(延迟地)填充缓存对象。 我提供了一
我正在试验 java.util.concurrent 并试图找出如何正确使用 AtomicReference.compareAndSet 来管理对单个共享状态单元的并发访问。 特别是:compareA
spring-data-redis 模块包含 RedisAtomicLong 类。 在这个类中你可以看到 public boolean compareAndSet(long expect, long
From the Java AtomicReferenceFieldUpdater docs : Note that the guarantees of the compareAndSet metho
源码是一样的。 public final boolean compareAndSet(V expect, V update) { return unsafe.compareAndSwapObj
在阅读 Java 中 java.util.Random 类的文档时,我偶然发现了 next 方法中的一些东西,我无法完全理解。 protected int next(int bits) { l
Java 的 AtomicInteger 提供 public final boolean compareAndSet(int expect, int update) .如果返回false,我想知道比较
线程标题应该是不言自明的......我对 AtomicBoolean 类的以下方法的规范有点困惑: java.util.concurrent.atomic.AtomicBoolean#compareA
我希望将 AtomicLong 中的当前 timestamp 与 currentTimeMS 进行比较,以便我知道是否已经过去了一段时间,如果是,则只有一个单线程将进入一个代码块,但据我所知,comp
来自AtomicLong的源代码: public final boolean compareAndSet(long expect, long update) { return
因为 Atomic 意味着线程安全。当 .set() 本身在 Java 中是原子和线程安全的时,我们什么时候使用 compareAndSet? 举例来说,我想自动设置一个变量,这样每个其他线程都可以看
static boolean unsynchronizedSetter(Date expected){ Date newDate = new Date(); AtomicReferen
VarHandle 显示以下错误 - Exception in thread "main" java.lang.NoSuchMethodError: VarHandle.compareAndSet(V
我正在实现一个请求实例的 FIFO 队列(为速度而预先分配的请求对象),并开始使用 add 方法上的“同步”关键字。该方法很短(检查固定大小缓冲区中是否有空间,然后将值添加到数组)。使用visualV
我想知道调用之间是否有任何区别(或可能的副作用): AtomicBoolean.set(true) 和 AtomicBoolean.compareAndset(false, true) AtomicB
我是一名优秀的程序员,十分优秀!