关于synchronized

首先看下下面这段代码:

public class Test{
     private static boolean stop;
     public static void main(String[] args){
          new Thread(new Runnable(){
                    public void run(){
                         int i = 0;
                         while(!stop){
                              i++;
                         }
                    }
               }).start();

          TimeUnit.SECONDS.sleep(1);
          stop = true;
     }
}

我们期望主线程在1秒之后,将stop设置为true,使后台线程停止运行。
但是,在某些机器上,后台线程永远在循环!

问题在于:stop变量在主线程和后台线程中都被使用,但是没有设置同步,虚拟机在优化代码的时候,会将:
                         while(!stop){
                              i++;
                         }

转变为:
                         if(!stop){
                              while(true){
                                    i++;
                              }
                         }


修正这个问题的一种方式是使用同步方式访问stop:
public class Test{
     private static boolean stop;
     private static synchronized void setStop(boolean s){
          stop = s;
     }
     private static synchronized boolean getStop(){
          return stop;
     }
     public static void main(String[] args){
          new Thread(new Runnable(){
                    public void run(){
                         int i = 0;
                         while(!getStop()){
                              i++;
                         }
                    }
               }).start();

          TimeUnit.SECONDS.sleep(1);
         setStop(true);
     }
}

总结:在多个线程访问同一个数据,即使这个数据是原子的,为了保证一个线程写入的值对另一个线程可见,同步是非常必要的。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值