在并发编程中,需要解决两个问题:线程间如何通信&线程间如何同步

线程同步:控制不同线程操作顺序的机制

解决这两个问题的方案有两种:共享内存&消息传递

  共享内存:通过使用共享内存,隐式通信和同步;这里程序员必须显式的指定某个方法或代码块要在线程间互斥执行

  消息传递:通过发消息来通信和同步;由于接收消息必须在发送消息之后,因此算是隐式的设置了同步

而JAVA采用的是共享内存模型。

JMM(Java内存模型)如下

如上图所示,JMM定义了线程和主内存之间的关系:线程之间的共享内存都存储在共享内存中,每个线程有自己的本地内存,本地内存存储了该线程以读/写共享变量的副本。

JMM是通过控制主内存与每个线程的本地内存之间的交互,来为java程序员提供内存可见性保证(例如有共享变量x=1,线程A和线程B都要修改这个共享变量,那么线程A将x设置为2,同时将x回写到共享内存,那么线程2拿到的就是x=2)

一、概念

说到JMM,就不得不提重排序、happends-before、as-if-serial

1、重排序

  处理器对代码的执行顺序并非按照程序编写的源代码顺序执行,而是编译器和处理器会对源代码做重排序,处理器按照重排序后的结果执行,从而提高性能。

  重排序分为三类:

    编译器优化的重排序:编译器在不改变单线程程序语义的情况下重排序

    指令并行重排序:如果不存在数据依赖,处理器可以重排序

    内存系统重排序:处理器使用缓存和读写缓存区,这使得加载和存储操作看起来乱序。

  从JAVA源代码到最终执行的指令序列所经历的重排序如下:

其中1是属于编译器的重排序,2和3属于处理器的重排序。

上述重排序中,对于内存系统的重排序,由于读写缓存只对当前线程可见,因此会造成多线程运行时的结果异常,因此处理器一般都只会允许对读写进行重排序。

2、happends-before

如果A happends-before于B,那么A的执行一定在B之前

3、as-if-serial

  不管怎么重排序,单线程的执行结果不能被改变。所有的重排序都要遵循这个原则。

二、内存语义

但是如果要是多线程运行,那么由于指令重排和线程的读写缓存问题,会造成执行结果异常,因此就需要使用volatile、锁、final来处理

1、volatile的内存语义

  可见性:对一个volatile修饰的变量的读,总能看到任意线程对这个volatile修饰的变量最后的写入

  原子性:对于任意由volatile修饰的变量的读和写操作都是原子操作;但是对于volatile++这样的操作是非原子性的。

  当写一个volatile修饰的变量时,JMM会把该线程对应的本地内存中的共享变量值刷新到内存中(只要有一个共享变量是volatile修饰,所有的共享变量都会被刷新到主内存中)

  当读一个volatile修饰的变量时,JMM会把该线程对应的内存置为无效,线程将从主内存读取共享变量。

2、锁的内存语义

  当锁释放时,JMM会把该线程对应的共享内存刷新到主内存中

  当获取锁时,JMM会把该线程对应的本地内存置为无效

  由以上两点可见,释放锁的内存语义和volatile的写拥有相同的内存语义,获取锁的内存语义和volatile的读有相同的内存语义。

3、final的内存语义

  写final重排序:禁止把final域的写重排序到构造函数之外。这样可以确保在每次引用为任意线程可见之前,对象的final域已经正确初始化。

  读final重排序:在一个线程中,初次读对象的引用和读该对象中的fianl域之间禁止重排序。

  为什么要将fianl的写限制在构造方法之内:比如一个线程看到一个整形的final域为0(还未初始化,默认值),过一段时间去读取时,发现已经变为1(被初始化完成后的值),因此会造成获取final修饰的值不一致的问题,为了修复该问题,限制了final修饰的写必须在构造方法之内。

三、happends-before

  happends-before是JMM最核心的概念,JMM为了平衡程序员(要求强内存模型,保证内存可见性)和处理器(要求若内存模型,处理器可以自行优化)的需求,设计了happends-before。

  1、happends-before定义:

    a、如果一个操作happends-before另一个操作,那么第一个操作的结果将对第二个操作结果可见,且第一个操作必须在第二个操作之前。

    b、两个操作之间存在happend-before关系,并不意味着java平台具体的实现必须按照happends-before的顺序来执行。如果重排序的结果与happends-before的执行结果一致,那么重排序并不非法。

  上述a是对程序员做出的承诺:如果A happends-before B,那么A的操作必须对B可见,且A的操作在B之前。

  b是对处理器的约束原则:只要不改变程序运行结果,编译器和处理器可以自行优化(这里的程序指的是单线程程序或正确同步的多线程程序)

  2、happends-before规则

    a、程序顺序规则:一个线程中的每个操作,happends-before于该线程任意的后续操作

    b、监视器锁规则:对于一个锁的解锁操作,happends-before于后续对该锁的加锁操作

    c、volatile规则:对于一个volatile修饰的写,hanppends-before于对这个volatile修饰的读

    d、传递性:如果A happends-before B,B happends-before C,那么A hanppends-before C

    e、start()规则:如果线程A中调用了线程B的start方法,那么线程A中调用B线程的start方法的操作happends-before于线程B的任何操作

    f、join()规则:如果线程A调用线程B的join方法,那么线程B中的任意操作happend-before于线程A从B.join()操作返回成功

四、双重检查锁定与延迟初始化

  对于很多场景,都会延迟加载类,来降低初始化类和创建对象的开销,待使用时在延迟加载。

  其实这里就是我们常说的单例模式。

  

public class InitDemo {
private static InitDemo instance; public static InitDemo getInstance(){
if(instance == null){
instance = new InitDemo();
}
return instance;
} public static InitDemo getInstance1(){
synchronized (InitDemo.class){
if(instance == null){
instance = new InitDemo();
}
}
return instance;
} public static InitDemo getInstance2(){
if(instance == null){
synchronized (InitDemo.class){
if(instance == null){
instance = new InitDemo();//有问题
}
}
}
return instance;
}
}

以上写法都是有问题的:

  getInstance方法的问题:非线程安全,可能会创建多个实例(俗称:懒汉式)

  getInstance1方法的问题:每次使用都加锁,性能消耗大

  getInstance2方法的问题:同样是非线程安全的

解决方案:

解决方案一:使用volatile

public class InitDemo {
private volatile static InitDemo instance; public static InitDemo getInstance(){
if(instance == null){
synchronized (InitDemo.class){
if(instance == null){
instance = new InitDemo();//有问题
}
}
}
return instance;
}
}

由于使用了volatile,因此多线程(都是第一次访问)创建对象时,就可以保证线程安全。

解决方案二:基于类初始化方式

public class InitDemo {
private static class InitDemoFactory{
public static InitDemo instance = new InitDemo();
} public static InitDemo getInstance(){
return InitDemoFactory.instance;
}
}

优缺点对比:

  基于类加载模式的方案,代码更简洁;但是只能对静态字段实现延迟加载

  使用volatile,除了可以对静态字段做延迟加载外,还可以对实例字段实现延迟加载。

并发02--JAVA内存模型的更多相关文章

  1. 高效并发一 Java内存模型与Java线程(绝对干货)

    高效并发一 Java内存模型与Java线程 本篇文章,首先了解虚拟机Java 内存模型的结构及操作,然后讲解原子性,可见性,有序性在 Java 内存模型中的体现,最后介绍先行发生原则的规则和使用. 在 ...

  2. Java并发编程-Java内存模型

    JVM内存结构与Java内存模型经常会混淆在一起,本文将对Java内存模型进行详细说明,并解释Java内存模型在线程通信方面起到的作用. 我们常说的JVM内存模式指的是JVM的内存分区:而Java内存 ...

  3. 【深入理解JAVA虚拟机】第5部分.高效并发.1.Java内存模型与线程。

    1.概述 摩尔定律:描述处理器晶体管数量与运行效率之间的发展关系.Amdahl定律:通过系统中并行化与串行化的比重来描述多处理器系统能获得的运算加速能力. 从摩尔定律到Amdahl定律的转变,代表了近 ...

  4. 02 | Java内存模型:看Java如何解决可见性和有序性问题

    什么是 Java 内存模型? 导致可见性的原因是缓存,导致有序性的原因是编译优化,那解决可见性. 有序性最直接的办法就是禁用缓存和编译优化,但是这样问题虽然解决了,我们程序的性能可就堪忧了.   合理 ...

  5. 并发编程-Java内存模型

    将之前看过的关于并发编程的东西总结记录一下,本文简单记录Java内存模型的相关知识. 1. 并发编程两个关键问题 并发编程中,需要处理两个关键问题:线程之间如何通信及线程之间如何同步. (1)在命令式 ...

  6. 02 java内存模型

    java内存模型 1.JVM内存区域 方法区:类信息.常量.static.JIT (信息共享) java堆:实例对象 GC (信息共享) OOM VM stack:JAVA方法在运行的内存模型 (OO ...

  7. 并发编程 —— Java 内存模型总结图

    关于 Java 内存模型的类似思维导图. 如有错误,还请指正.

  8. 【死磕Java并发】-----Java内存模型之happend-before

    在上篇博客([死磕Java并发]-–深入分析volatile的实现原理)LZ提到过由于存在线程本地内存和主内存的原因,再加上重排序,会导致多线程环境下存在可见性的问题.那么我们正确使用同步.锁的情况下 ...

  9. 【死磕Java并发】-----Java内存模型之happens-before

    在上篇博客([死磕Java并发]-–深入分析volatile的实现原理)LZ提到过由于存在线程本地内存和主内存的原因,再加上重排序,会导致多线程环境下存在可见性的问题.那么我们正确使用同步.锁的情况下 ...

  10. 【死磕Java并发】-----Java内存模型之重排序

    在执行程序时,为了提供性能,处理器和编译器常常会对指令进行重排序,但是不能随意重排序,不是你想怎么排序就怎么排序,它需要满足以下两个条件: 在单线程环境下不能改变程序运行的结果: 存在数据依赖关系的不 ...

随机推荐

  1. WEB前端程序员需要的网站整理

    前端学习资源实在是又多又广,在这样的一个知识的海洋里,我们像一块海绵一样吸收,想要快速提高效率,平时的总结不可缺少,以下总结了一些,排版自我感觉良好,推送出来. 一.插件类网站 jQuery插件库:h ...

  2. Java实现 LeetCode 494 目标和

    494. 目标和 给定一个非负整数数组,a1, a2, -, an, 和一个目标数,S.现在你有两个符号 + 和 -.对于数组中的任意一个整数,你都可以从 + 或 -中选择一个符号添加在前面. 返回可 ...

  3. Java实现最近点问题

    **问题描述:** 给定某空间中(直线空间或平面空间)n个点,请找出它们中的最近点对.你需要完成下列任务: 1.随机产生或手工输入n个点的坐标. 2.输出最近的两个点的坐标. 3.算法尽可能效率高. ...

  4. java实现子集和问题

    1 问题描述 求n个正整数构成的一个给定集合A = {a1,a2,a3,-,an}的子集,子集的和要等于一个给定的正整数d.请输出所有符合条件的子集. 2 解决方案 2.1 全排列思想求解 方法1:首 ...

  5. java实现第五届蓝桥杯出栈次序

    出栈次序 X星球特别讲究秩序,所有道路都是单行线.一个甲壳虫车队,共16辆车,按照编号先后发车,夹在其它车流中,缓缓前行. 路边有个死胡同,只能容一辆车通过,是临时的检查站,如图[p1.png]所示. ...

  6. Java学习之第二天

    一.流程控制 1.顺序结构:自上而下,依次执行(从上到下,一直走下去) 2.选择结构:(1)if .if—else.嵌套if (2)switch(mod){ case 1:执行代码 case 2:执行 ...

  7. 大话微服务(Big Talk in MicroService)

    下面开始分析我的microservice 之旅. what? 是什么 why? 为什么 how? 什么做 1.什么是微服务 microservice 是 SOA(Service-Oriented Ar ...

  8. Sublime Text 3.1 注册码

    加入到hosts文件: 127.0.0.1 www.sublimetext.com 127.0.0.1 license.sublimehq.com hosts 文件的位置: Windows : c:/ ...

  9. SpringBoot设置mysql的ssl连接

    因工作需要,mysql连接需要开启ssl认证,本文主要讲述客户端如何配置ssl连接. 开发环境信息: SpringBoot: 2.0.5.RELEASE mysql-connector-java: 8 ...

  10. 别人开发三年30k,而我才12K,看完他面试前狂刷的面试题,我惊了

    朋友做Java开发三年多的时间了,在老东家勤勤恳恳工作了三年多,工资也就是从刚开始的8K涨到了12K,天天给我吐槽他的工资低.2019年中下旬开始就一直在各种地方找资源,刷面试题,想要“骑驴找马”,所 ...