Java线程安全性知识总结-0

线程安全性:
当多个线程访问某个类时,不管运行时环境采用何种调度方式或者这些进程将如何交替执行,并且在主调代码中不需要任何额外的同步或者协同,这个类都能表现出正确的行为,那么就称这个类是线程安全的。

线程安全性表现在3个方面:

1.原子性:提供了互斥访问,同一时刻只能有一个线程来对它进行操作。

Java中最常见就是AtomicXXX:CAS、Unsafe.compareAndSwapInt。

image.png

对于AtomicLong,JDK1.8有更多的解决方案,也就是LongAdder类。

CAS也是有适用场景的,比如资源竞争小,是非常适用的,不用进行内核态和用户态之间的线程上下文切换的,同时自旋概率也会大大减少,提升性能。但是资源竞争激烈时,比如大量线程对同一资源进行写和读操作,这样CAS就不适用了。因为在这种情况下自旋概率会大大增加,从而浪费CPU资源,降低性能。

对于long和double变量,JVM会将64位的读操作和写操作拆成2个32位的操作。LongAdder的实现思想:热点数据分离。把AtomicLong内部核心数据value分离成一个数组,每个线程访问时通过hash算法映射到其中一个数字进行计数。最终计数的结果就是数组的求和累加。热点数据value会被分离成多个Cell。每个Cell独立维护着内部的值。当前对象实际的值由所有Cell累计求和,这样热点得到了有效的分离,提高了并行度。相对于AtomicLong,LongAdder将单点的压力分离到各个节点上。在低并发压力下的时候,通过对base的直接更新,可以很好的保障和AtomicLong的性能保持一致。

LongAdder在统计的时候如果有并发更新,可能会导致统计的数据有误差。

CAS中最著名的ABA问题,可以通过版本号来解决。也就是AtomicStampedReference这个类,版本号对应着里面的stamp变量。


2.可见性:一个线程对主内存的修改可以及时的被其他线程观察到。导致共享变量在线程间不可见的原因:线程交叉执行、重排序结合线程交叉执行、共享变量更新后的值没有在工作内存与主存间及时更新。

在我之前的文章Java底层知识总结-0有提到过JMM中同步规则。

对于可见性,JVM提供了Volatile和Synchronized。

JMM关于synchronized的两条规定:

  • 线程解锁前,必须要把共享变量的最新值刷新到主内存中。

  • 线程加锁时,将清空工作内存中共享变量的值,从而使用共享变量时需要从主内存中重新读取最新的值。(注意,加锁和解锁是同一把锁)

volatile是通过加入内存屏障和禁止指令重排序优化来实现的。

内存屏障也称之为内存栅栏或者栅栏指令,是一种屏障指令。它使CPU或者编译器对屏障指令之前和之后发出的内存操作执行一个排序约束。这通常意味着在屏障之前发布的操作被保证在屏障之后发布的操作之前执行。

Load Load屏障:Load1和Load2代表2条读取指令。在Load2要读取的数据被访问前要保证Load1要读取的数据已经被读取完毕。

StoreStore屏障:Store1和Store2代表2条写入指令。在Store2写入执行之前,要保证Store1的写入操作对其他处理器可见。

LoadStore屏障:在Store2被写入前,要保证Load1要读取的数据被读取完毕。

StoreLoad屏障:在Load2读取操作执行前,要保证Store1的写入对所有处理器可见。StoreLoad屏障的开销是4种屏障中最大的。

happens-before是JSR-133规范。内存屏障是CPU指令,可以简单认为前者是最终目的,后者是实现手段。

在一个变量被volatile修饰后,JVM会做2件事情:

  • 在volatile变量写入操作之前加入StoreStore屏障(将工作内存中的共享变量刷新到主内存),在写入操作之后加入StoreLoad屏障。

  • 在volatile变量读操作之后加入LoadLoad屏障(从主内存读取共享变量),在读操作之后加入LoadStore屏障。

volatile适用于以下情况:

  • 运行结果并不依赖变量的当前值,或者能够确保只有单一的线程修改变量的值。
  • 变量不需要与其他的状态变量一起参与不变约束。

3.有序性:一个线程观察其他线程中的指令执行顺序,由于指令重排序的存在,该观察结果一般杂乱无序。

Java内存模型中,允许编译器和处理器对指令进行重排序,但是重排序过程不会影响到单线程程序的执行,却会影响到多线程并发执行的正确性。

能够通过volatile、synchronized、Lock手段达到有序性。

JMM模型具备一些先天的有序性,不需要任何手段达到有序性。happen-before的先行发生原则如下:

  • 程序次序规则:一个线程内,按照代码程序,书写在前面的操作先行发生于书写在后面的操作。

  • 锁定规则:一个unlock操作先行发生于后面对同一个锁的lock操作。

  • volatile变量原则:对一个变量的写操作先行发生于后面对这个变量的读操作。

  • 传递规则:如果操作A先行发生于操作B,而操作B又先发生于操作C,则可以得出操作A先行发生于操作C。

  • 线程启动规则:Thread对象的start()方法先行发生于此线程的每一个动作。

  • 线程中断原则:对线程interrupt()方法的调用先行于被中断线程的代码检测到中断事件的发生。

  • 线程终结规则:线程中所有的操作都先行发生于线程的终止检测,我们可以通过Thread.join()方法结束,Thread.isAlive()的返回值手段检测到线程已经终止执行。

  • 对象终结规则:一个对象的初始化完成先行发生于它的finalize()方法的开始。

如果2个操作的执行次序无法从happen-before的原则推导出来,就无法保证它们之间的有序性。JVM就可以随意的对它们进行指令重排序。

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

推荐阅读更多精彩内容