Java之多线程内存可见性_2(volatile不能保证原子性)

来源:互联网 发布:淘宝代销商品怎么下单 编辑:程序博客网 时间:2024/03/28 20:41

以下内容在慕课网-->细说Java多线程之内存可见性

volatile不能保证原子性


public class VolatileDemo{private volatile int number = 0;public int getNumber(){return this.number;}public void increase(){this.number++;}public static void main(String[], args){final VolatileDemo volDemo = new VolatileDemo();for(int i=0; i<500;i++){new thread(new Runnable(){public void run(){volDemo.increase();}}).start();}//如果还有子线程在运行,主线陈就让出cpu资源,//知道所有的子线程都运行完了,主线程再继续往下执行while(Thread.activiCount() > 1){Thread.yield();}System.out.println("number:" + volDemo.getNumber());}}




实际执行结果不一定是500
因为:不能保证原子性,可能多个线程交叉执行

public void increase(){this.number++;}


number++  不是原子性
如:
number = 5
1.线程A读取number的值
2.线程B读取number的值
3.线程B执行加1操作
4.线程B写入最新的number的值
主内存:number = 6
线程B工作内存:number = 6
线程A工作内存:number = 5
5.线程A执行加1操作
6.线程A写入最新的number值,主内存的number = 6
7.整个过程2次number++实际上只加了1

解决方案:
保证number自增操作的原子性:
1.使用synchronized关键字
2.使用ReentrantLock(java.until.concurrent.locks包下, jdk1.5)
3.使用AtomicInterger(vava.util.concurrent.atomic包下, jdk1.5)


使用synchronized解决
修改:
1.
private volatile int number = 0; 修改为private int number = 0;


2.
public void increase(){synchronized(this){this.number++;}}



使用ReentrantLock解决 : 
修改:
1.增加  
private Lock lock = new ReentrantLock();

2.
public void increase(){lock.lock();//加锁try{   //推荐写法   this.number++;}finally{//锁内部操作可能会抛出一些异常,所以保证锁一定能被释放   lock.unlock();//解锁 } }

volatile适合场合
要在多线程中安全的使用volatile变量,必须同时满足”

1.对变量的写入操作不依赖其当前值(改变后的值不能与之前的值有关系)
如:不满足:number++  count = count*5等
满足:boolean变量、记录温度变化的变量等

2.该变量没有包含在具有其他变量的不变式中(有多个volatile变量,每个volatile变量状态独立于其他volatile变量)
如:程序中有2个volatile变量low,up 
不满足:不变式low<up

0 0
原创粉丝点击