028-86922220

建站动态

根据您的个性需求进行定制 先人一步 抢占小程序红利时代

Java中怎么实现多线程的可见性与有序性

本篇文章给大家分享的是有关Java中怎么实现多线程的可见性与有序性,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

龙口ssl适用于网站、小程序/APP、API接口等需要进行数据传输应用场景,ssl证书未来市场广阔!成为成都创新互联公司的ssl证书销售渠道,可以享受市场价格4-6折优惠!如果有意向欢迎电话联系或者加微信:13518219792(备注:SSL证书合作)期待与您的合作!

多线程的可见性

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

共享变量

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

Java内存模型

JMM(Java Memory Model,简称JMM)描述了Java程序中各种变量(线程共享变量)的访问规则,以及在JVM中将变量存储到内存和从内存中读取出变量这样的底层细节。它遵循四个原则:

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

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

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

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

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

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

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

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

Java的内存模型: Java中怎么实现多线程的可见性与有序性

Java语言层面支持的可见性实现方式有以下两种:

  1. synchronized

  2. volatile

synchronized

JMM关于synchronized的规定:

因此线程执行synchronized代码执行互斥锁的过程:

  1. 获得互斥锁。

  2. 清空工作内存。

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

  4. 执行代码

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

  6. 释放互斥锁

synchronize在JDK6之后,进行了很多优化。更多synchroinzed的讲解,点击查看:详解Java多线程锁之synchronized

volatile

不能保证原子性,但适合使用volatile修饰状态标记量

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

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

volatile写操作: Java中怎么实现多线程的可见性与有序性

volatile读操作: Java中怎么实现多线程的可见性与有序性

多线程的有序性

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

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

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

  2. 指令集并行重排序(处理器优化)

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

happens-before原则

JMM可以通过happens-before关系向程序员提供跨线程的内存可见性保证(如果A线程的写操作a与B线程的读操作b之间存在happens-before关系,尽管a操作和b操作在不同的线程中执行,但JMM向程序员保证a操作将对b操作可见)。

以上就是Java中怎么实现多线程的可见性与有序性,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注创新互联行业资讯频道。


标题名称:Java中怎么实现多线程的可见性与有序性
网站链接:http://www.tsicrk.com/article/pochcs.html

其他资讯

让你的专属顾问为你服务

1.6431s