java volidate线程安全_03.(多线程与并发)面试题-02--Volidate的原理和指令重排序

线程栈(线程的工作内存)保存了线程运行时候变量值信息。当线程访问某一个对象时候值的时候,首先通过对象的引用找到对应在堆内存的变量的值,然后把堆内存变量的具体值load到线程本地内存中,建立一个变量副本,之后线程就不再和对象在堆内存变量值有任何关系,而是直接修改副本变量的值,

在修改完之后的某一个时刻(线程退出之前),自动把线程变量本的值回写到对象在堆中变量。这样在堆中的对象的值就产生变化了。下面一幅图副描述这写交互

f636512ac26e5cd83ae56b6e21ca04c2.png

read and load 从主存复制变量到当前工作内存 use 代码中使用值

assign  改变共享变量值  store and write 用工作内存数据刷新主存相关内容

其中use and assign 可以多次出现

但是这一些操作并不是原子性,也就是 在read load之后,如果主内存count变量发生修改之后,线程工作内存中的值由于已经加载,不会产生对应的变化,所以计算出来的结果会和预期不一样

总结:关键字上使用voildate之后,每一次use之前一定要read和load,这样就保证了可见性,其他线程

指令重排序

总结:jvm会对指令执行的顺序进行优化,这样是为了提高执行的效率。但是在单线程的情况下指令的执行先后没有关系,但是在多线程的情况下这些指令的执行顺序就是对其他线程产生很大的影响。

很多介绍JVM并发的书或文章都会谈到JVM为了优化性能,采用了指令重排序,但是对于什么是指令重排序,为什么重排序会优化性能却很少有提及,其实道理很简单,假设有这么两个共享变量a和b:

private int a;

private int b;

在线程A中有两条语句对这两个共享变量进行赋值操作:

a = 1;

b = 2;

假设当线程A对a进行复制操作的时候发现这个变量在主内存已经被其它的线程加了访问锁,那么此时线程A怎么办?等待释放锁?不,等待太浪费时间了,它会去尝试进行b的赋值操作,b这时候没被人占用,因此就会先为b赋值,再去为a赋值,那么执行的顺序就变成了:

b = 2;

a = 1;

对于在同一个线程内,这样的改变是不会对逻辑产生影响的,但是在多线程的情况下指令重排序会带来问题,看下面这个情景:

在线程A中:

context = loadContext();

inited = true;

在线程B中:

while(!inited ){

sleep

}

doSomethingwithconfig(context);

假设A中发生了重排序:

inited = true;

context = loadContext();

那么B中很可能就会拿到一个尚未初始化或尚未初始化完成的context,从而引发程序错误。

想到有一条古老的原则很适合用在这个地方,那就是先要保证程序的正确然后再去优化性能。此处由于重排序产生的错误显然要比重排序带来的性能优化要重要的多。要解决重排序问题还是通过volatile关键字,volatile关键字能确保变量在线程中的操作不会被重排序而是按照代码中规定的顺序进行访问,同时使用synchronized 关键字,里面也不会进行指令重排序。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436243.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

选址问题java_学习使用分治算法来解决邮局选址问题(Java实现)

邮局选址问题(分治算法)前言提示:在算法的学习过程中我们会遇到各种各样的算法思想,其中最常见的就有分治算法思想,而本文的问题邮局选址问题就是基于分治思想而去实现的一个日常问题提示:以下是本文内容,我将对该问题…

新学期新开始

新学期,新开始!告诉自己,一定要坚持走自己的路。太多的事情,太多的诱惑。我们要坚持自己的脚步,要知道:坚持就是胜利。终将会有自己的一片天地。不能被中间的事情分神,忘了自己远方的那盏明灯&a…

210板wince键盘驱动分析和移植

主要是bsp包里面的3个cpp文件的分析。 首先是pddlist.cpp文件 这个文件介绍了3种键盘的wince的api入口函数名前面加winapi),本文只关心第二种矩阵键盘。 然后在kbd.cpp文件中可以找到这个入口函数Matrix_Entry。 这里值得注意的是,本驱动…

【转】各种消息下wParam及lParam值的含义

01、WM_PAINT消息 LOWORD(lParam)是客户区的宽,HIWORD(lParam)是客户区的高 02、滚动条WM_VSCROLL或WM_HSCROLL消息 LOWORD(wParam)指出了鼠标对滚动条的操作。比如上、下、左、右、翻页、移动等。 03、击键消息,有WM_SYSKEYDOWN、WM_SYSKEYUP、WM_KEY…

WinAPI: GetCurrentThread、GetCurrentThreadId、GetCurrentProcess、GetCurrentProcessId

{返回当前线程的虚拟句柄} GetCurrentThread: THandle;{返回当前线程 ID} GetCurrentThreadId: DWORD;{返回当前进程的虚拟句柄} GetCurrentProcess: THandle;{返回当前进程 ID} GetCurrentProcessId: DWORD;提示:ID 是系统唯一的标识.所谓虚拟句柄, 就是该句柄只在调用进程的进…

Windows Embedded CE 6.0开发初体验(一)Windows CE概述

这篇文章的目的并不是介绍Windows Embedded CE开发的方方面面,只是用一个初涉嵌入式领域的软件开发者的视角来介绍Windows CE开发中最常用的概念和知识,解决大家会在实际开发中碰到的一些小问题,让大家了解嵌入式开发与应用程序开发&#xff…

【转】C++学习一 结构与结构体

转自&#xff1a;https://www.cnblogs.com/fantianliang/category/1574590.html https://www.cnblogs.com/fantianliang/p/11620328.html 结构体属于C与C区别之一。 代码例子如下&#xff08;来自《C程序设计》&#xff09;&#xff1a; #include <iostream> #includ…

Windows Embedded CE 6.0开发初体验(二)CE开发环境

CE开发环境 下面就要让今天的主角闪亮登场了——Digi CC 9P开发板。为了感谢Digi提供的开发板&#xff0c;还是给他们做个广告吧&#xff1a;这是一家以WiFi相关产品为主营业务的公司&#xff0c;也生产ARM架构的CPU&#xff0c;所以做开发板也是他们公司份内的事情了。总的感…

请问 CType和DirectCast函数有何区别

DirectCast 关键字引入类型转换操作。该关键字的使用方法与 CType 关键字相同&#xff0c;如下列所示&#xff1a; Dim Q As Object 2.37 Requires Option Strict to be Off. Dim I As Integer CType(Q, Integer) Succeeds. Dim J As Integer DirectCast(Q, Integer) …

【转】C++学习二 vector的用法(使用sort对于vector排序)

一、vector的介绍 vector是C里面的一个容器&#xff0c;也是我们数学上面理解的向量&#xff0c;有一些比较常见的操作。 二、vector的定义 #include<vector> using namespace std;//int 说明的是vector存储的数据类型 vector<int> odd_vector; 三、vector的一些…

java syncr_JAVA基础—Synchronized线程同步机制

Synchronize的使用场景Synchronize可以使用在一下三种场景&#xff0c;对应不同的锁对象场景synchronized代码块synchronized方法synchronized静态方法锁对象任意对象this 对象该类的字节码对象 *.class没有同步代码实例public class SyncSample {public static void main(Stri…

Windows Embedded CE 6.0开发初体验(三)设置Boot-loader

在Digi CC 9P开发板中已经预置了Boot-loader和一个CE 6.0的系统&#xff0c;所以&#xff0c;我们直接启动开发板&#xff0c;就可以运行一个CE 6.0的操作系统。不过&#xff0c;我们的最终目的是要开发一个自己的CE 6.0平台&#xff0c;所以我们需要对Boot-loader中的一些系统…

【转】C++学习三 模板类出错总结(Missing template arguments before ‘L‘)

一、模板类的说明 模板类有一个好处是可以放宽你输入的数据类型。 比如有这样的一个函数&#xff1a; int add(int x, int y) {return xy; } 这个函数对于int类型的x,y才适合&#xff0c;但是如果我们希望计算float类型的呢&#xff1f; 这必须重新定义一个函数&#xff…

Flex 学习站点汇总,(FLEX学习站点、博客、论坛)

来自&#xff1a;http://www.zhuoqun.net/article.asp?id501不错的整理&#xff01;感谢原作者。今天把收藏夹共享出来&#xff0c;希望对学习Flex的人有所帮助。一、国外站点1.资源类Adobe Flex 2 Component Explorer&#xff1a; 官方的&#xff0c;展示了各种组件的用法&am…

java 延迟实例化_延迟初始化Spring Bean的几种方式

XML 配置:Java 注解:Lazy(true)Spring 中默认是非延迟加载Bean的,也就是提前把Bean初始化好,用的时候直接用. 优点是运行的时候比较快(提前初始化了,直接用). 缺点是启动慢和占用内存,因为要初始化很多Bean.延迟加载是需要的时候再去初始化Bean. 优点是解约内存,启动快(不需要提…

Windows Embedded CE 6.0开发初体验(四)跑个应用程序先

因为我们的开发板上已经预置了Windows CE 6.0的系统&#xff0c;所以我们可以先创建一个.NET CF的应用程序&#xff0c;来看一下如何将.NET CF程序部署到Digi开发板上。 第一步&#xff0c;首先在Visual Studio 2005中创建一个新的工程。这一步和开发普通的Windows Mobile应用…

【转】C++学习四 冒泡排序法的一些改进

冒泡排序法需要两次扫描&#xff0c;所以从时间复杂度来说&#xff0c;是O(n2). 如果用图形表示&#xff0c;是这样的&#xff1a; 但是我们可以加以改进。 首先是&#xff0c;如果在排序中间&#xff0c;整个向量已经达到了有序状态&#xff0c;可以直接跳出来。 这样它的复…

java中的jgroup_JGroups实现聊天小程序

本文实例为大家分享了JGroups实现聊天小程序的具体代码&#xff0c;供大家参考&#xff0c;具体内容如下效果图&#xff1a;代码部分&#xff1a;package com.lei.jgoups;import java.io.BufferedReader;import java.io.DataInputStream;import java.io.DataOutputStream;impor…

Windows Embedded CE 6.0开发初体验(五)构建CE平台

接下来的一节会有非常多的图&#xff0c;因为我们要在Visual Studio 2005里创建一个CE 6.0的平台&#xff08;Platform&#xff09;了。创建CE平台的方法和创建普通工程的方法类似。在完成了Platform Builder之后&#xff0c;我们打开Visual Studio 2005后&#xff0c;选择File…

est.java 2 错误 找不到符号_找不到Cython/Python符号PyString\u Typ

我试图调用Cython的C代码&#xff0c;并遵循official guide。在但是&#xff0c;当在python中导入已编译的模块时&#xff0c;我得到以下错误。在---------------------------------------------------------------------------ImportError Traceback (most recent call last)i…