java - AQS中源码疑问
问题描述
在AbstractQueuedSynchronizer类中维护了一个用volatile修饰的state状态,而这个状态有如下的两种修改方法:
state的set方法:
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操作能保证线程安全呢?还是我的理解有问题呢?谢谢各位大牛了!
问题解答
回答1:并不冲突吧,setState只是一个写操作,并没有管原来的state,比如state原本是0,线程1和线程2分别读取了这个值0,线程1把它set成1,然后线程2打算把它set成2,是可以成功的,尽管state已经是1了,而且线程2也知道,但是无所谓,线程2只是把它set成2。下面那个方法就不会了吧。
回答2:是的,你的理解有问题。
不是说volatile修饰的变量在多线程的单操作中,能够保证其写后读的可见性,即能保证线程安全
能够保证可见性,不意味着可以保证线程安全。可见性跟线程安全不是同一个概念。
cas操作,可以划分为几个小操作
比较 expect 和 state 变量当前的值,如果相同,继续2,如果不同,方法结束。
为 state 赋值 update
这两个操作,如果是多线程并发调用,是会有线程安全问题的。这里的 cas 方法利用了 cpu 的 cas 指令,这个指令是原子操作。可以避免并发问题。
回答3:简而言之:
不依赖原始值的可以使用set
依赖原始值的可以使用cas去设置。本身这是个乐观锁。
相关文章:
1. MYSQL新建用户设置可以远程访问的问题2. python - 求一个在def中可以实现调用本def满足特定条件continue效果的方法(标题说不太清楚,请见题内描述)3. java - mybatis怎么实现在数据库中有就修改,没有就添加4. $fields = $values = [];这条代码一直定义不了,一直报错,老师的源码也是被报错的,执行不了,请问该怎么解决这个问题5. node.js - nodejs和前端JavaScript 字符串处理结果不一样是什么原因?6. 各位谁知道这个CSS时候哪里出错了???7. mysql - 我的myeclipse一直连显示数据库连接失败,不知道为什么8. mysql 5.7单表300万数据,性能严重下降,如何破?9. mysql - SQL操作时间的函数?10. mysql 为何insert的时候会有lock wait timeout 异常
![$fields = $values = [];这条代码一直定义不了,一直报错,老师的源码也是被报错的,执行不了,请问该怎么解决这个问题](http://www.haobala.com/attached/image/news/202205/093622cb60.png)