多线程系列第(五)篇---synchronized和volatile

1.问题描述

假设银行有个电子账户,专门用来统计存款总额,存款来源于很多个存储用户。

代码示例

public class VolatileDemo {
private  volatile static int number;  //volatile 不能保证原子性

private static void add(){
    number++;
}

public static void main(String[] args) {
    
    for(int i=0;i<1000;i++){
        new Thread(new Runnable() {
            
            public void run() {
                add();
            }
        }).start();
    }
    
    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    System.out.println(number);  //输出结果可能是999
    //因为num++不是原子性操作,可以分解为三个步骤,读取num的值,+1,写入工作内存
  }
}

运行结果
1000或者999

程序分析
1.number 表示银行的电子账户金额
2.for循环的1000个线程代表一个储户
3.add代表存钱操作,假设每次存1块钱
4.主线程在睡眠了2秒后再去查金额,就是为了让所有的线程都运行完毕,再去查询。

2.问题产生的原因

表面上看是多线程操作产生的问题,要分析深层次的原因,需要先知道以下几个知识点

2.1 java内存模型

1.所有共享变量都存储在主内存中
2.每个线程都有自己独立的工作内存,里面保存该线程使用到的变量的副本(主内存中该变量的一份拷贝)
3.不同的线程无法访问对方工作内存中的变量,线程之间的传值需要通过主内存来实现

2.2 原子性和可见性

原子性
原子是世界上最小的单位,具有不可分割的性质,原子性说的是某个操作具有不可分割性,如a=0,有的操作是可以被分割的,如a++,它实际上是a=a+1,可以分为三个操作,取值,+1,将值写入工作内存

可见性
可见性说的是线程之间的可见性,一个线程修改的状态对另一个线程是可见的。也就是一个线程修改的结果,另一个线程马上就能看到

2.3 原因分析

上述程序电子账户金额number开始时,我并没有加入volatile 关键字

原因一:number的不可见性
由于每个线程修改的只是工作内存中的值,可能没有及时将工作内存中的值刷新到主内存中
比如线程1读取到num值为99,加1后变成100,这时候这个值并没有及时刷新到主内存中,线程2读取到的值仍然是99,加1后num的值还是100

解决办法,想办法使得number具有可见性,这时就会联想到volatile。volatile变量在每次被线程访问时,都强迫从主内存中重读该变量的值,当该变量发生变化时,又会强迫线程将该变量的值刷新到主内存中
但是加上之后,仍然会得到999的结果,仔细看add的代码,发现number++,这个操做不具有原子性

原因二:number++不具有原子性
number++,在计算机指令中可以分为三个操作

  1. 读取number的值
  2. 将number加1
  3. 将number的值写入工作内存中

假设线程A,执行操作1,读取到num的值为99,此时被线程B占用运行权,线程B读取num的值为99,继续执行操作2和3,num的值变为100,线程B执行结束,运行权重回线程A,虽然主内存中num的值已经是100了,但是线程A的操作1已经在之前就做了,这时会直接执行2和3,执行结束后num的值仍然为100,结果当然会出现问题

结论:必须要同时保证原子性和可见性才不会出现上面的问题,这就引出了今天我要介绍的主角之一:synchronized

3. synchronized

3.1 volatile 和 synchronized 比较
  • volatile用来修饰变量,synchronized用来修饰方法或代码块
  • volatile本质是告诉jvm当前变量在寄存器中是不确定的,需要从主内存中读取,synchronized是对对象加锁,从而保证synchronized块同一时间,只能由一个线程访问
  • volatile只能保证可见性,而synchronized则既可以保证变量修改的可见性,也能保证代码块的原子性
  • volatile不会造成线程阻塞,而synchronized会造成线程阻塞,即争夺锁的使用权
3.2 synchronized代码示例
public class SynchronizedDemo {
private static int number;

private synchronized static void add() {
    number++;
}

public static void main(String[] args) {

    for (int i = 0; i < 1000; i++) {
        new Thread(new Runnable() {

            public void run() {
                add();
            }
        }).start();
    }

    try {
        Thread.sleep(2000);
    } catch (InterruptedException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    System.out.println(number);
  }
}

运行结果
1000,运行了20多次,最上面的基本运行个10几次就会出现999的结果
3.3 synchronized其他一些需要注意的地方
  • synchronized(xx.class)与synchronized(this)的区别
    前者锁的是该类的所有实例对象,后者锁的是类对象(类的其中一个对象)。
    若类对象被锁,则类对象的所有同步方法全被锁;
    若实例对象被锁,则该实例对象的所有同步方法全被锁

  • synchronized methods(){} 与synchronized(this){}
    synchronized methods(){} 与synchronized(this){}之间没有什么区别。
    只是synchronized methods(){} 便于阅读理解,而synchronized(this){}可以更精确的控制冲突限制访问区域,有时候表现更高效率。

  • synchronized(xx.class) 等价于 synchronized static methods(){}

  • synchronized(this) 等价于 synchronized methods(){}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,406评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,732评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,711评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,380评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,432评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,301评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,145评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,008评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,443评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,649评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,795评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,501评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,119评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,731评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,865评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,899评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,724评论 2 354

推荐阅读更多精彩内容