`

细说Java多线程之内存可见性

 
阅读更多

http://www.imooc.com/learn/352

细说Java多线程之内存可见性

 

一、可见性

可见性:一个线程对共享变量值的修改,能够及时地被其他线程看到。

共享变量:如果一个变量在多个线程的工作内存中都存在副本,那么这个变量就是这几个线程的共享变量

 

二、Java内存模型(JMM)

JAVA内存模型(Java Memory Model)描述了Java程序中各种变量(线程共享变量)的访问规则,以及在JVM中将变量存储到内存和从内存中读取出变量这样的底层细节。

1、所有的变量都存储在主内存中

2、每个线程都有自己独立的工作内存,里面保存该线程使用到的变量的副本(主内存中该变量的一份拷贝)



 

 



 

 

三、两条规定

1、线程对共享变量的所有操作都必须在自己的工作内存中进行,不能直接从主内存中读写

2、不同线程之间无法直接访问其他线程工作内存中的变量,线程间变量值的传递需要通过主内存来完成。

 

四、共享变量可见性实现的原理

线程1对共享变量的修改要想被线程2及时看到,必须要经过如下2个步骤:

1、把工作内存1中更新过的共享变量刷新到主内存中

2、将主内存中最新的共享变量的值更新到工作内存2中



 

 



 

 



 

 

五、可见性

要实现共享变量的可见性,必须保证两点:

1、线程修改后的共享变量值能够及时从工作内存刷新到主内存中

2、其他线程能够及时把共享变量的最新值从主内存更新到自己的工作内存中

 

六、可见性的实现方式

Java语言层面支持的可见性实现方式:

1、synchronized

2、volatile

 

七、synchronized实现可见性

synchronized能够实现:

1、原子性(同步)

2、可见性

 

JMM关于synchronized的两条规定:

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

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

 

线程解锁前对共享变量的修改在下次加锁时对其他线程可见

 

线程执行互斥代码的过程:

1、获得互斥锁



 

 

2、清空工作内存

 

3、从主内存拷贝变量的最新副本到工作内存



 

 

4、执行代码



 

 

5、将更改后的共享变量的值刷新到主内存



  

 

6、释放互斥锁

 

重排序

重排序:代码书写的顺序与实际执行的顺序不同,指令重排序是编译器或处理器为了提高程序性能而做的优化

1、编译器优化的重排序(编译器优化)

2、指令级并行重排序(处理器优化)

3、内存系统的重排序(处理器优化)



 

 

as-if-serial

as-if-serial:无论如何重排序,程序执行的结果应该与代码顺序执行的结果一致(Java编译器、运行时和处理器都会保证Java在单线程下遵循as-if-serial语义)



 

 

可见性分析

 

导致共享变量在线程间不可见的原因:

1、线程的交叉执行

2、重排序结合线程交叉执行

3、共享变量更新后的值没有在工作内存与主内存间及时更新

 

 

不可见的原因:              synchronized解决方案:

1、线程的交叉执行-->原子性

2、重排序结合线程交叉执行-->原子性

3、共享变量未及时更新-->可见性

 

synchronized实现可见性代码:

package com;

public class SynchronizedDemo {

	//共享变量
	private boolean ready = false;
	private int result = 0;
	private int number = 1;
	
	//写操作
	public void write(){
		ready = true;	//1.1
		number = 2;		//1.2
	}
	
	//读操作
	public void read(){
		if(ready){				//2.1
			result = number*3;	//2.2
		}
		System.out.println("result的值为:"+result);
	}
	
	
	//内存线程类
	private class ReadWriteThread extends Thread {
		//根据构造方法中传入的flag参数,确定线程执行读操作还是写操作
		private boolean flag;
		public ReadWriteThread(boolean flag){
			this.flag = flag;
		}
		@Override
		public void run(){
			if(flag){
				//构造方法中传入true,执行写操作
				write();
			}else{
				//构造方法中传入false,执行读操作
				read();
			}
		}
	}
	
	public static void main(String[] args) {
		SynchronizedDemo sysDemo = new SynchronizedDemo();
		//启动线程执行写操作
		sysDemo.new ReadWriteThread(true).start();
		try {
			Thread.sleep(1000);
		} catch (InterruptedException e) {
			e.printStackTrace();
		}
		//启动线程执行读操作
		sysDemo.new ReadWriteThread(false).start();
	}
	
}

  

 

 

八、Volatile实现可见性

volatile关键字:

1、能够保证volatile变量的可见性

2、不能保证volatile变量复合操作的原子性

 

Volatile如何实现内存可见性:

深入来说:通过加入内存屏障和禁止重排序优化来实现的。

1、对volatile变量执行写操作时,会在写操作后加入一条store屏障指令。

2、对volatile变量执行读操作时,会在读操作前加入一条load屏障指令。

 

通俗地讲:volatile变量在每次被线程访问时,都强迫从主内存中重读该变量的值,而当该变量发生变化时,又会强迫线程将最新的值刷新到主内存。这样任何时刻,不同的线程总能看到该变量的最新值。

 

线程写volatile变量的过程:

1、改变线程工作内存中volatile变量副本的值

2、将改变后的副本的值从工作内存刷新到主内存

 

 

线程写volatile变量的过程:

1、从主内存中读取volatile变量的最新值到线程的工作内存中。

2、从工作内存中读取volatile变量的副本

 

volatile实现可见性代码:

package com;

public class VolatileDemo {
	private volatile int number = 0;
	public int getNumber() {
		return this.number;
	}
	public void increase() {
		this.number++;
	}
	public static void main(String[] args) {
		VolatileDemo volatileDemo = new VolatileDemo();
		for (int i = 0; i < 500; i++) {
			new Thread(new Runnable() {
				@Override
				public void run() {
					volatileDemo.increase();
				}
			}).start();
		}
		// 如果还有子线程在运行,主线程就让出CPU资源
		// 直到所有的子线程都运行完了,主线程再继续往下执行
		while (Thread.activeCount() > 1) {
			Thread.yield();
		}
		System.out.println("number:" + volatileDemo.getNumber());
	}
}

 

 

 

volatile不能保证volatile变量复合操作的原子性:

private int number = 0;

number ++ ; 不是原子操作

1、读取number的值

2、将number的值加1

3、写入最新的number的值

 

synchronized(this){

number++;

}

 

private volatile int number = 0;

变为volatile变量,无法保证原子性;

 

 

程序分析

number=5

1、线程A读取number的值

2、线程B读取number的值

3、线程B执行加1操作

4、线程B写入最新的number的值

5、线程A执行加1操作

6、线程A写入最新的number值

 

两次number++,只增加1次

 

解决方案:

保证number自增操作的原子性:

1、使用synchronized关键字

2、使用ReentrantLock(java.until.concurrent.locks包下)

3、使用AtomicInterger(java.until.concurrent.atomic包下)

 

 

volatile适合场合

要在多线程中安全的使用volatile变量,必须同时满足:

1、对变量的写入操作不依赖其当前值

不满足:number++、count=count*5等

满足:boolean变量、记录温度变化的变量等

 

2、该变量没有包含在具有其他变量的不变式中

不满足:不变式 low<up

 

synchronized和volatile比较

1、volatile不需要加锁,比synchronized更轻量级,不会阻塞线程;

2、从内存可见性角度讲,volatile读相当于加锁,volatile写相当于解锁。

3、synchronized既能保证可见性,又能保证原子性,而volatile只能保证可见性,无法保证原子性。

 

 

总积

 

1、什么是内存可见性

2、Java内存模型(JMM)

3、实现可见性的方式:synchronized和volatile

4、synchronized和volatile实现内存可见性的原理

5、synchronized实现可见性

指令重排序

as-if-serial

6、volatile实现可见性

volatile能够保证可见性

volatile不能保证原子性

volatile使用注意事项

 

7、synchronized和volatile比较

volatile比synchronized更轻量级

volatile没有synchronized使用的广泛

 

 

问:即时没有保证可见性的措施,很多时候共享变量依然能够在主内存和工作内存见得到及时地更新?

 

 

答:一般只有在短时间内高并发的情况下才会出现变量得不到及时更新的情况,因为CPU在执行时会很快地刷新缓存,所以一般情况下很难看到这种问题。

 

对64位(long、double)变量的读写可能不是原子操作:

1、Java内存模型允许JVM将没有被volatile修饰的64位数据类型的读写操作划分为两次32的读写操作来进行

 

导致问题:有可能会出现读取到“半个变量”的情况

解决方法:加volatile关键字

 

 

  • 大小: 202.4 KB
  • 大小: 237.7 KB
  • 大小: 142.7 KB
  • 大小: 161.4 KB
  • 大小: 166.7 KB
  • 大小: 108.7 KB
  • 大小: 33.1 KB
  • 大小: 30.5 KB
  • 大小: 50 KB
  • 大小: 131.9 KB
  • 大小: 268 KB
分享到:
评论

相关推荐

    C#_细说多线程(上下)

    ### C# 多线程详解 #### 一、线程的定义 1.1 **进程、应用程序域与线程的关系** - **进程**:在Windows系统中,进程是最基本的概念之一,代表一个运行中的程序实例。每个进程都有独立的地址空间和其他资源。进程...

    细说java线程问题

    ### 细说Java线程问题:深入理解与实践 在Java编程中,线程是执行的基本单位,它允许程序在操作系统层面并发执行多个任务。掌握Java线程的原理及其使用方法,对于提升软件性能和响应性至关重要。本文将从线程的概念...

    细说Java NIO

    3. **选择器(Selector)**:选择器允许单线程处理多个通道,提高了程序的并行性。通过注册感兴趣的事件(如读、写、连接和接受),可以选择器监控多个通道的状态。 **二、Java NIO的主要特性** 1. **非阻塞**:在...

    细说Java之util类.

    例如,ArrayList和HashMap不是线程安全的,如果在多线程环境中同时访问,需要自己添加同步控制。而Vector和Hashtable虽然提供了线程安全,但由于其同步机制过于简单,可能会导致性能问题。 在实际编程中,根据需求...

    C#综合揭秘--细说多线程

    《C#综合揭秘--细说多线程》 在计算机科学中,多线程是一种并发执行的任务处理方式,它允许程序同时执行多个独立的代码片段,从而提高资源利用率和程序性能。C#作为.NET框架的主要编程语言,提供了丰富的多线程支持...

    C#综合揭秘_细说多线程.pdf

    本文主要从线程的基础用法,CLR线程池当中工作者线程与I/O线程的开发,并行操作PLINQ等多个方面介绍多线程的开发。 其中委托的BeginInvoke方法以及回调函数最为常用。 而 I/O线程可能容易遭到大家的忽略,其实在开发...

    细说Java之常用集合类.rar

    最后,关于学习资源,"细说Java之常用集合类.pdf"这本书籍教程应该详细介绍了这些集合类的用法和实现细节,可以帮助开发者更好地理解和运用Java集合框架。"下载说明.txt"可能包含获取更多资源的指南,如"A5下载- 更...

    细说Java之常用集合类

    ### 细说Java之常用集合类 #### 一、引言 在Java开发过程中,我们经常需要处理数据的集合操作,比如存储、检索、排序等。为了方便开发者使用,Java SDK提供了一系列内置的集合类,它们主要位于`java.util`包中。...

    细说UI线程和Windows消息队列

    ### 细说UI线程与Windows消息队列 在探讨Windows应用程序的工作原理时,理解UI线程和消息队列的概念至关重要。UI线程(User Interface Thread)与消息队列(Message Queue)是实现用户交互的核心机制,它们确保了应用...

    细说Windows XP 虚拟内存优化.doc

    ### 细说Windows XP 虚拟内存优化 #### 一、引言 虚拟内存是在操作系统层面为了提高系统的运行效率而引入的一个概念。它能够帮助操作系统有效地管理内存资源,尤其是在物理内存不足的情况下,通过虚拟内存可以提升...

    C#综合揭秘——细说多线程(中)

    本文主要从线程的基础用法,CLR线程池当中工作者线程与I/O线程的开发,并行操作PLINQ等多个方面介绍多线程的开发。一、线程的定义二、线程的基础知识三、以ThreadStart方式实现多线程四、CLR线程池的工作者线程五、...

    细说Android 4.0 NDK编程 源码

    线程同步则确保了多线程环境下的数据安全,防止竞态条件。错误处理则要求程序员能够妥善处理异常情况,确保程序的健壮性。 书中的源码分析部分将有助于读者更直观地理解这些概念。通过实际的代码示例,读者可以学习...

    韩顺平细说servlet.CreateCode.java

    韩顺平细说servlet.生成验证码的代码

    韩顺平细说Servlet源代码

    此外,还会涉及Servlet的多线程模型,讲解为什么Servlet通常是线程安全的,以及如何处理并发访问。 除此之外,课程还会涵盖Servlet的会话管理,包括`HttpSession`对象的使用,以及如何跟踪用户的会话状态。另外,还...

    《细说PHP》

    它的特点包括跨平台性、面向对象、易于学习和丰富的函数库等。 在学习PHP时,我们会遇到各种语法和结构。比如变量声明、基本的数据类型、运算符、流程控制(包括条件语句和循环语句)、数组、函数等。这些基础知识...

    细说UI线程和Windows消息队列.doc

    消息队列的存在保证了线程的同步和异步操作的有序性,使得即使在多线程环境中,用户界面也能保持一致和稳定。 消息的生命周期始于用户的输入操作,如键盘敲击或鼠标点击,操作系统捕获这些事件后生成消息,将其放入...

    C#综合揭秘——细说多线程(上)

    本文主要从线程的基础用法,CLR线程池当中工作者线程与I/O线程的开发,并行操作PLINQ等多个方面介绍多线程的开发。其中委托的BeginInvoke方法以及回调函数最为常用。而I/O线程可能容易遭到大家的忽略,其实在开发多...

Global site tag (gtag.js) - Google Analytics