黄色网址大全免费-黄色网址你懂得-黄色网址你懂的-黄色网址有那些-免费超爽视频-免费大片黄国产在线观看

Java面向?qū)ο?/div>
Java異常
Java數(shù)組
Java常用類
Java集合
Java IO流
Java線程
Java反射
Socket編程
Java注解開發(fā)
Java GoF設(shè)計模式
HashMap
Java內(nèi)存模型
Java線性表

Java內(nèi)存模型的概念

volatile這個關(guān)鍵字可能很多朋友都聽說過,或許也都用過。在Java 5之前,它是一個備受爭議的關(guān)鍵字,因為在程序中使用它往往會導(dǎo)致出人意料的結(jié)果。在Java 5之后,volatile關(guān)鍵字才得以重獲生機。

volatile關(guān)鍵字雖然從字面上理解起來比較簡單,但是要用好不是一件容易的事情。由于volatile關(guān)鍵字是與Java的內(nèi)存模型有關(guān)的,因此在講述volatile關(guān)鍵之前,我們先來了解一下與內(nèi)存模型相關(guān)的概念和知識,然后分析了volatile關(guān)鍵字的實現(xiàn)原理,最后給出了幾個使用volatile關(guān)鍵字的場景。

以下是本文的目錄大綱:

一、Java內(nèi)存模型的概念

二、Java并發(fā)編程中的三個概念

三、Java內(nèi)存模型

四、深入剖析volatile關(guān)鍵字

五、使用volatile關(guān)鍵字的場景

Java內(nèi)存模型的概念

大家都知道,計算機在執(zhí)行程序時,每條指令都是在CPU中執(zhí)行的,而執(zhí)行指令過程中,勢必涉及到數(shù)據(jù)的讀取和寫入。由于程序運行過程中的臨時數(shù)據(jù)是存放在主存(物理內(nèi)存)當(dāng)中的,這時就存在一個問題,由于CPU執(zhí)行速度很快,而從內(nèi)存讀取數(shù)據(jù)和向內(nèi)存寫入數(shù)據(jù)的過程跟CPU執(zhí)行指令的速度比起來要慢的多,因此如果任何時候?qū)?shù)據(jù)的操作都要通過和內(nèi)存的交互來進行,會大大降低指令執(zhí)行的速度。因此在CPU里面就有了高速緩存。

也就是,當(dāng)程序在運行過程中,會將運算需要的數(shù)據(jù)從主存復(fù)制一份到CPU的高速緩存當(dāng)中,那么CPU進行計算時就可以直接從它的高速緩存讀取數(shù)據(jù)和向其中寫入數(shù)據(jù),當(dāng)運算結(jié)束之后,再將高速緩存中的數(shù)據(jù)刷新到主存當(dāng)中。舉個簡單的例子,比如下面的這段代碼:

i = i + 1;

當(dāng)線程執(zhí)行這個語句時,會先從主存當(dāng)中讀取i的值,然后復(fù)制一份到高速緩存當(dāng)中,然后CPU執(zhí)行指令對i進行加1操作,然后將數(shù)據(jù)寫入高速緩存,最后將高速緩存中i最新的值刷新到主存當(dāng)中。

這個代碼在單線程中運行是沒有任何問題的,但是在多線程中運行就會有問題了。在多核CPU中,每條線程可能運行于不同的CPU中,因此每個線程運行時有自己的高速緩存(對單核CPU來說,其實也會出現(xiàn)這種問題,只不過是以線程調(diào)度的形式來分別執(zhí)行的)。本文我們以多核CPU為例。

比如同時有2個線程執(zhí)行這段代碼,假如初始時i的值為0,那么我們希望兩個線程執(zhí)行完之后i的值變?yōu)?。但是事實會是這樣嗎?

可能存在下面一種情況:初始時,兩個線程分別讀取i的值存入各自所在的CPU的高速緩存當(dāng)中,然后線程1進行加1操作,然后把i的最新值1寫入到內(nèi)存。此時線程2的高速緩存當(dāng)中i的值還是0,進行加1操作之后,i的值為1,然后線程2把i的值寫入內(nèi)存。

最終結(jié)果i的值是1,而不是2。這就是著名的緩存一致性問題。通常稱這種被多個線程訪問的變量為共享變量。

也就是說,如果一個變量在多個CPU中都存在緩存(一般在多線程編程時才會出現(xiàn)),那么就可能存在緩存不一致的問題。

為了解決緩存不一致性問題,通常來說有以下2種解決方法:

1、通過在總線加LOCK#鎖的方式

2、通過緩存一致性協(xié)議

這2種方式都是硬件層面上提供的方式。

在早期的CPU當(dāng)中,是通過在總線上加LOCK#鎖的形式來解決緩存不一致的問題。因為CPU和其他部件進行通信都是通過總線來進行的,如果對總線加LOCK#鎖的話,也就是說阻塞了其他CPU對其他部件訪問(如內(nèi)存),從而使得只能有一個CPU能使用這個變量的內(nèi)存。比如上面例子中 如果一個線程在執(zhí)行 i = i +1,如果在執(zhí)行這段代碼的過程中,在總線上發(fā)出了LCOK#鎖的信號,那么只有等待這段代碼完全執(zhí)行完畢之后,其他CPU才能從變量i所在的內(nèi)存讀取變量,然后進行相應(yīng)的操作。這樣就解決了緩存不一致的問題。

但是上面的方式會有一個問題,由于在鎖住總線期間,其他CPU無法訪問內(nèi)存,導(dǎo)致效率低下。所以就出現(xiàn)了緩存一致性協(xié)議。最出名的就是Intel 的MESI協(xié)議,MESI協(xié)議保證了每個緩存中使用的共享變量的副本是一致的。它核心的思想是:當(dāng)CPU寫數(shù)據(jù)時,如果發(fā)現(xiàn)操作的變量是共享變量,即在其他CPU中也存在該變量的副本,會發(fā)出信號通知其他CPU將該變量的緩存行置為無效狀態(tài),因此當(dāng)其他CPU需要讀取這個變量時,發(fā)現(xiàn)自己緩存中緩存該變量的緩存行是無效的,那么它就會從內(nèi)存重新讀取。

主站蜘蛛池模板: 日本69xxx18hd| 精品国产免费一区二区三区 | 啪啪免费网站入口链接 | 久久久午夜影院 | 手机看片日韩高清国产欧美 | 国产欧美日韩免费 | 国产123区在线视频观看 | 成人深爱网 | 国产成人综合亚洲 | 热re66久久精品国产99re | 国产精品亚洲精品观看不卡 | 国产一区二区三区免费在线视频 | 国产成在线观看免费视频 | 日本r级视频 | 欧美理论片在线观看一区二区 | 久久艹精品 | 黄色福利片 | 香蕉大伊亚洲人在线观看 | 91视频第一页 | 国产在线欧美精品中文一区 | 看真人一级毛多毛片 | 在线成人毛片 | 黑丝在线观看 | 污污视频在线观看黄 | 2020亚洲欧美日韩在线观看 | 虫族bl文全肉高h | 午夜一级片 | 中文有码视频 | 一级做a爰性色毛片免费 | 理论福利午夜 | 亚洲一区二区免费在线观看 | 人人玩人人添人人澡免费 | 天堂资源在线中文 | 91精品国产免费久久久久久青草 | 国产三级欧美 | 最近高清中文字幕大全免费1 | 国产亚洲精品日韩香蕉网 | 麻豆成人久久精品二区三区小说 | 国产第一福利136视频导航 | 欧美一级免费在线观看 | 一个人看的高清www视频免费 |