Java并发总结

JMM内存抽象

JMM定义了线程和主内存之间的抽象关系:线程之间的共享变量存储在主内存中,每个线程都有一个私有的本地内存,本地内存中存储了该线程以读/写共享变量的副本。
本地内存是JMM的一个抽象概念,并不真实存在。它涵盖了缓存,写缓冲区,寄存器以及其他的硬件和编译器优化。
同时JMM确保在不同的编译器和不同的处理器平台之上,通过禁止特定类型的编译器重排序和处理器重排序,为程序员提供一致的内存可见性保证

所有实例域、静态域和数组元素存储在堆内存中,堆内存在线程之间共享
局部变量,方法定义参数和异常处理器参数不会在线程之间共享,它们不会有内存可见性问题,也不受内存模型的影响

重排序

在执行程序时为了提高性能,编译器和处理器常常会对指令做重排序

  1. 编译器优化的重排序:编译器在不改变单线程程序语义的前提下,可以重新安排语句的执行顺序
  2. 指令级并行的重排序:现代处理器采用了指令级并行技术(Instruction-Level Parallelism, ILP)来将多条指令重叠执行。如果不存在数据依赖性,处理器可以改变语句对应机器指令的执行顺序
  3. 内存系统的重排序:由于处理器使用缓存和读/写缓冲区,这使得加载和存储操作看上去可能是在乱序执行
  • 对于编译器:JMM的编译器重排序规则会禁止特定类型的编译器重排序
  • 对于处理器重排序:JMM的处理器重排序规则会要求java编译器在生成指令序列时,插入特定类型的内存屏障(memory barriers)指令,通过内存屏障指令来禁止特定类型的处理器重排序

内存屏障

为了保证内存可见性,java编译器在生成指令序列的适当位置会插入内存屏障指令来禁止特定类型的处理器重排序

屏障类型 指令示例 说明
LoadLoad Barriers Load1; LoadLoad; Load2 确保Load1数据的装载,之前于Load2及所有后续装载指令的装载
StoreStore Barriers Store1; StoreStore; Store2 确保Store1数据对其他处理器可见(刷新到内存),之前于Store2及所有后续存储指令的存储
LoadStore Barriers Load1; LoadStore; Store2 确保Load1数据装载,之前于Store2及所有后续的存储指令刷新到内存
StoreLoad Barriers Store1; StoreLoad; Load2 确保Store1数据对其他处理器变得可见(指刷新到内存),之前于Load2及所有后续装载指令的装载。StoreLoad Barriers会使该屏障之前的所有内存访问指令(存储和装载指令)完成之后,才执行该屏障之后的内存访问指令

StoreLoad Barriers是一个“全能型”的屏障,它同时具有其他三个屏障的效果。执行该屏障开销会很昂贵,因为当前处理器通常要把写缓冲区中的数据全部刷新到内存中(buffer fully flush)

Happen-Before原则

JMM通过这个概念来阐述操作之间的内存可见性:如果一个操作执行的结果需要对另一个操作可见,那么这两个操作之间必须存在happens-before关系
两个操作既可以是在一个线程之内,也可以是在不同线程之间

  • 程序顺序规则:一个线程中的每个操作,happens- before 于该线程中的任意后续操作
  • 监视器锁规则:对一个监视器锁的解锁,happens- before 于随后对这个监视器锁的加锁
  • volatile变量规则:对一个volatile域的写,happens- before 于任意后续对这个volatile域的读
  • 传递性:如果A happens- before B,且B happens- before C,那么A happens- before C

注意,两个操作之间具有happens-before关系,并不意味着前一个操作必须要在后一个操作之前执行!happens-before仅仅要求前一个操作(执行的结果)对后一个操作可见,且前一个操作按顺序排在第二个操作之前(the first is visible to and ordered before the second)

并发层次

通过Volatile机制和CAS指令实现了多核的

缓存一致性

通过MESI协议实现多核间的缓存一致性
CPU所有的操作都是在其各自核的缓存内完成的,如果不回写,其他核根本不知道变量被修改了,只有当CPU对缓存进行回写并通过MESI协议通知时才知道

Volatile

轻量级的同步机制,不会引起线程的上下文切换
当写一个volatile变量时,JMM会把该线程对应的本地内存中的共享变量刷新到主内存
当读一个volatile变量时,JMM会把该线程对应的本地内存置为无效。线程接下来将从主内存中读取共享变量。即每次都从主存中读取最新的数据
但Volatile只能保证可见性,不能保证原子性

CAS

保证多核下指令执行的原子性,但操作的是主存,直接绕过了缓存(不会触发缓存回写)
所以要实现多核间的可见性,CAS操作的变量要使用volatile修饰

原子变量实现

利用volatile和cas
以AtomicInteger的实现为例

private volatile int value;

public final int incrementAndGet() {
  // 一直循环,直到更新成功为止
  for (;;) {
    // 获取当前值
    int current = get();
    int next = current + 1;
    // 使用cas原子性更新
    if (compareAndSet(current, next))
      return next;
    }
}

// volatile修饰,每次都读取最新的值
public final int get() {
  return value;
}

AQS实现

本质就是状态+等待链表
通过一个内部的Int变量来代表状态,状态的值代表通过还是等待
如果等待,则提供一个非阻塞的等待链表(通过CAS来实现节点的增删),同时在释放时,唤醒等待的线程
AQS是一个基础类,所有需要阻塞等待的,都可以使用其特性来实现

锁实现

继承于AQS,对于互斥锁,状态0为则没有通过,并设置为1,否则阻塞等待

public void lock() {
  sync.lock();
}

// NonFair为例
final void lock() {
  // 设置状态成功(0变成1),成功获取锁
  if (compareAndSetState(0, 1))
    setExclusiveOwnerThread(Thread.currentThread());
  else
    // 再尝试申请,失败则等待
    acquire(1);
}

// AQS
public final void acquire(int arg) {
  // 如果尝试申请失败,则将当前线程加入到等待队列中
  if (!tryAcquire(arg) && acquireQueued(addWaiter(Node.EXCLUSIVE), arg))
    selfInterrupt();
}

final boolean nonfairTryAcquire(int acquires) {
  final Thread current = Thread.currentThread();
  // 再次尝试申请一次
  int c = getState();
  if (c == 0) {
    if (compareAndSetState(0, acquires)) {
      setExclusiveOwnerThread(current);
      return true;
    }
  }
  // 如果是当前线程获取的锁,则直接增加次数即可
  else if (current == getExclusiveOwnerThread()) {
    int nextc = c + acquires;
    if (nextc < 0) // overflow
      throw new Error("Maximum lock count exceeded");
    setState(nextc);
    return true;
  }
  return false;
}

并发数据结构

这里不对每个数据结构的实现方式做详细描述,只简单地列一下每个数据结构的大概使用方式

数据结构 说明
SynchronousQueue 并不是一个队列,可以看成size=1的队列,一次只能放入一个元素
PriorityBlockingQueue 优先级队列
LinkedBlockingQueue 链表队列,队列为空时需要等待,适合生产者-消费者模型
LinkedBlockingDeque 链表的双端队列
DelayQueue 延迟队列
CopyOnWriteArraySet 修改时进行复制的集合
CopyOnWriteArrayList 修改时进行复制的队列
ConcurrentSkipListSet 有序的Set
ConcurrentSkipListMap 有序的Map
ConcurrentLinkedQueue 并发队列,添加、获取都不会等待
ConcurrentHashMap 并发HashMap
ArrayBlockingQueue 数组队列,有界

参考

深入理解Java内存模型(一)——基础
深入理解Java内存模型(四)——volatile
聊聊并发(一)——深入分析Volatile的实现原理
深入理解Java内存模型(五)——锁

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

推荐阅读更多精彩内容