volatile 功能作用
1、保证可见性;
public class ThreadVolatileKeJian {
volatile int i = 0; //volatile 加了 与不加 while (tv.i0) 状态来展示 可见性
public void sum60(){
this.i=60;
}
public static void main(String[] args) {
ThreadVolatileKeJian tv = new ThreadVolatileKeJian();
new Thread(()->{
try {
TimeUnit.SECONDS.sleep(3);
} catch (InterruptedException e) {
e.printStackTrace();
}
tv.sum60();
System.out.println(Thread.currentThread().getName()+","+tv.i);
},“aaaaaa”).start();
while (tv.i0){
}
System.out.println(Thread.currentThread().getName()+","+ tv.i);
}
}
2、禁止指令重排序;
不能保证原子性???
实验
//证明 volatile 不能保证原子性
public class ThreanVolatileNoAtomic {
volatile int i = 0;
public void sum60(){
this.i=60;
}
public void isum(){
this.i++;
}
public static void main(String[] args) {
ThreanVolatileNoAtomic tv = new ThreanVolatileNoAtomic();
for(int i=0;i<=20;i++){
new Thread(()->{
for(int j=0;j<=1000;j++){
tv.isum();
}
},String.valueOf(i)).start();
}
//等待上面所有线程全部执行完成以后,在main线程查看最终结果
while (Thread.activeCount()>2){
//暂停当前正在执行的线程对象,并执行其他线程。
Thread.yield();
}
System.out.println(Thread.currentThread().getName()+","+ tv.i);
}
}
输出结果都不等于 20000
如何才能让 volatile 保证原子性呢?
1、可以使用自带原子性的类 atomic
为什么atomic 类能 解决 volatile 不能保证原则性的问题呢?
CAS自旋锁
https://blog.csdn.net/weixin_45673552/article/details/102761789 (CAS自旋锁原理)