AQS中源码疑问

发布于 2022-09-04 14:35:30 字数 700 浏览 11 评论 0

AbstractQueuedSynchronizer类中维护了一个用volatile修饰的state状态,而这个状态有如下的两种修改方法:

  • stateset方法:

protected final void setState(int newState) {
    state = newState;
}
  • CAS方法:

protected final boolean compareAndSetState(int expect, int update) {
    // See below for intrinsics setup to support this
    return unsafe.compareAndSwapInt(this, stateOffset, expect, update);
}

那么,我的疑问来了,不是说volatile修饰的变量在多线程的单操作中,能够保证其写后读的可见性,即能保证线程安全,为什么还提供了CAS操作能保证线程安全呢?还是我的理解有问题呢?谢谢各位大牛了!

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(3

安人多梦 2022-09-11 14:35:30

并不冲突吧,setState只是一个写操作,并没有管原来的state,比如state原本是0,线程1和线程2分别读取了这个值0,线程1把它set成1,然后线程2打算把它set成2,是可以成功的,尽管state已经是1了,而且线程2也知道,但是无所谓,线程2只是把它set成2。下面那个方法就不会了吧。

我的鱼塘能养鲲 2022-09-11 14:35:30

是的,你的理解有问题。

不是说volatile修饰的变量在多线程的单操作中,能够保证其写后读的可见性,即能保证线程安全

能够保证可见性,不意味着可以保证线程安全。可见性跟线程安全不是同一个概念。

cas操作,可以划分为几个小操作

  1. 比较 expect 和 state 变量当前的值,如果相同,继续2,如果不同,方法结束。

  2. 为 state 赋值 update

这两个操作,如果是多线程并发调用,是会有线程安全问题的。
这里的 cas 方法利用了 cpu 的 cas 指令,这个指令是原子操作。可以避免并发问题。

千纸鹤 2022-09-11 14:35:30

简而言之:

  1. 不依赖原始值的可以使用set

  2. 依赖原始值的可以使用cas去设置。本身这是个乐观锁。

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文