当前位置: 首页 > 知识库问答 >
问题:

compareAndSet()无法按预期工作

邓高韵
2023-03-14

我编写了自己的AtomicDouble类,还有一个BankAccount类,它执行两个简单的取款和存款操作,它有一个AtomicDouble实例(余额)。我的代码的问题是,当我在deposit()中调用addAndGet方法时,程序会陷入一个无限循环,compareAndSet()永远不会返回真值,但当我调试它时,currentValue和atomic中的值。get()相等,但此方法无法理解。

有趣的是,当我将if(atomic.get()==currentValue)替换为if(atomic.compareAndSet(currentValue,nextValue))时,程序运行正常。

public class AtomicDouble extends Number {

private final AtomicReference<Double> atomic;

public AtomicDouble() {
    this(0.0);
}

public AtomicDouble(double initialValue) {
    atomic = new AtomicReference<>(initialValue);
}

public final double addAndGet(double delta) {
    while (true) {
        double currentValue = atomic.get();
        double nextValue = currentValue + delta;
        if (atomic.compareAndSet(currentValue, nextValue))
            return nextValue;
    }
}

public final double incrementAndGet() {
    return addAndGet(1);
}

public final void set(double newValue) {
    atomic.set(newValue);
}

public final double get() {
    return atomic.get();
}


public final double getAndSet(double newValue) {
    return atomic.getAndSet(newValue);
}


public float floatValue() {
    return (float) get();
}

@Override
public double doubleValue() {
    return get();
}

public int intValue() {
    return (int) get();
}

public long longValue() {
    return (long) get();
}

public String toString() {
    return Double.toString(get());
}

}

public class BankAccount {
private final AtomicDouble balance;
private String accountNumber;

public BankAccount(double balance, String accountNumber) {
    this.balance = new AtomicDouble(balance);
    this.accountNumber = accountNumber;

}

public void deposit(double number, String color) {
    System.out.println(color + "deposit  " + number + " current balance=" + balance.addAndGet(number));
}

public void withdraw(double number, String color) {
    if (this.balance.get() - number >= 0) {
        System.out.println(color + "Withdraw " + number + " current balance=" + balance.addAndGet(-number));
        return;
    }
    System.out.println(color + "Not enough balance");
}


public static void main(String[] args) {

    BankAccount bankAccount = new BankAccount(1000.0, "4234236");

    ExecutorService threadsPool = Executors.newFixedThreadPool(2);
    threadsPool.execute(new Runnable() {
        @Override
        public void run() {
            bankAccount.deposit(300.0, ThreadColor.ANSI_YELLOW);
            bankAccount.withdraw(50.0, ThreadColor.ANSI_YELLOW);
        }
    });

    threadsPool.execute(new Runnable() {
        @Override
        public void run() {
            bankAccount.deposit(203.75, ThreadColor.ANSI_BLUE);
            bankAccount.withdraw(100.0, ThreadColor.ANSI_BLUE);
        }
    });

    threadsPool.shutdown();
}

}

输出:没有输出

共有1个答案

齐阳
2023-03-14

我想这是因为自动装箱。你不能有一个引用,你有一个引用

操作数每次在循环中都会被“重新装箱”,因此引用永远不会相同。也就是说,当前值中的引用永远不会与原子中的引用相同。

尝试使用currentValue引用类型。

public final double addAndGet(double delta) {
    while (true) {
        Double currentValue = atomic.get();
        Double nextValue = currentValue + delta;
        if (atomic.compareAndSet(currentValue, nextValue))
            return nextValue;
    }
}

(幸运的是,双是一个不可变的类型,否则这将有一个种族危险)

 类似资料:
  • 问题内容: 我正在使用selenium来抓取一些数据。 我单击的页面上有一个按钮,说“ custom_cols”。此按钮为我打开一个窗口,从中可以选择列。 此新窗口有时需要一些时间才能打开(大约5秒钟)。所以我已经使用了 延迟为20秒。但是有时它无法在新窗口中选择查找元素,即使该元素可见。在其余时间中,这种情况仅发生十次一次。 我在其他地方也使用了相同的功能(WebDriverWait),并且可以

  • 问题内容: 经过测试后,我只能对已经解析过的JSON数据返回一个肯定值。 根据官方文件: isValidJSONObject返回一个布尔值,该布尔值指示是否可以将给定对象转换为JSON数据。 但是,尽管事实是我尝试将其从JSON转换为NSDictionary的对象都可以正常转换,但仍会返回。 这是我的代码: 我的日志包含以下内容: 然后是dict的输出,这是一个巨大的NSMutableDictio

  • 问题内容: 考虑以下可以在任何程序执行之前预加载的库: 问题是,尽管总是调用全局变量的构造函数,但对于某些程序却不调用析构函数,例如: 对于其他一些程序,按预期方式调用析构函数: 您能解释一下为什么在第一种情况下不调用析构函数吗?编辑:上面的问题已得到解答,即程序可能会使用_exit(),abort()退出。 然而: 有没有办法在预加载的程序退出时强制调用给定函数? 问题答案: 具有作为其初始化代

  • 我必须将日期-时间字符串转换为分区日期-时间对象。我使用DateTimeForman读取模式。根据留档,模式中的“Z”可以接受以下格式: /-0000 但是“分区约会”。parse(myDate,formatter)只适用于第一种情况;相反,在第二种情况下,代码生成一个异常。 我用的是8Java 我做错什么了?谢谢!

  • 问题内容: 我是python新手,我熟悉循环并尝试了一本书中的示例 但是输出如下 问题答案: 您必须改用(Python 2.x),因为它等效于,因此它会将输入解析并评估为有效的Python表达式。 注意: 不会捕获用户错误(例如,如果用户输入了一些无效的Python表达式)。可以这样做,因为它将输入转换为。有关更多信息,请阅读Python docs 。

  • 问题内容: 在一个安静的星期六晚上解决一些拼图问题(wooohoo …没问题),并在sort()上挣扎。结果并不完全符合我的预期。该程序从100到999的每个组合中进行迭代,并检查产品是否为回文。如果是,请追加到列表。我需要排序的列表:D这是我的程序: 哪些网: 显然索引0大于1。知道发生了什么吗?我觉得它与尾随/前导零有关,但是我快速浏览了一下,但看不到问题所在。 奖励积分,如果您知道谜题来自哪