深入理解java虛擬機(6)---記憶體模型與線程 & Volatile

来源:http://www.cnblogs.com/deman/archive/2016/05/26/5531321.html
-Advertisement-
Play Games

其實關於線程的使用,之前已經寫過博客講解過這部分的內容: http://www.cnblogs.com/deman/category/621531.html JVM裡面關於多線程的部分,主要是多線程是如何實現的,以及高效併發。 1.Java記憶體模型 CPU在運行的時候,不可能把所有的東西都放在寄存器 ...


其實關於線程的使用,之前已經寫過博客講解過這部分的內容:

http://www.cnblogs.com/deman/category/621531.html

JVM裡面關於多線程的部分,主要是多線程是如何實現的,以及高效併發。

1.Java記憶體模型

CPU在運行的時候,不可能把所有的東西都放在寄存器裡面,所有需要使用記憶體。這個記憶體就是我們知道的那個記憶體。

但是實際情況是,記憶體的讀寫速度於CPU的指令操作差了幾個數量級。所以為了跟高效的使用CPU,就有高速緩存這麼一個東西。

以下是Intel 酷睿i7 6700K參數:

三級緩存8MB

百度以下就知道這個“三級緩存”是個神馬東西。

而java的記憶體模型與物理結構非常相識,有一個主記憶體,對應我們電腦的記憶體,還有每個線程都有一個工作記憶體,對應於高速緩存。

 

可以看到,每個java線程都有自己獨立的記憶體。

這也就解釋了,為什麼不同線程,如果不同步的話,變數就會有併發的問題。

這裡關於工作記憶體和主記憶體的拷貝問題,是由JVM實現的,並不是正真意義上的記憶體複製。

2.記憶體間操作

1)lock,作用於主記憶體變數,把一個變數標記為線程獨占。

2)unlock,與lock正相反。

3)read,作用於主記憶體變數,它把一個變數從主記憶體傳輸到工作記憶體中。

4)load,作用於工作記憶體變數,把從read裡面獲取的變數放入工作記憶體的變數副本中。

5)use,作用於工作記憶體變數,把變數的值傳遞給執行引擎。

6)assign,作用於工作記憶體變數,把執行引擎的值 複製給工作記憶體變數。同use相反

7)store,作用於工作記憶體變數,把工作記憶體變數傳輸到主記憶體中。

8)write,作用於主記憶體變數,把store獲取的值,寫入到住記憶體中的變數。

read & load, store & write成對出現。

還有其他一些規則,目的就是保證記憶體變數的 操作合理,有序。

3.併發編程的三個概念

1)原子性

計一個操作要麼全部執行,要麼不執行,不能被打斷。

jvm通過lock & unlock指令來保證代碼的原子性。反映到java代碼就是synchronized.

2)可見性

可見性是指當一個程式修改變數以後,其他程式可以立即獲得這個修改的值。

3)有序性

JVM在編譯java代碼,優化的時候,會重現排布java代碼的順序。但是會保證結果時候java代碼的順序結果一致的。

public Runnable mRun1 = new Runnable() {
@Override
public void run() {
int a = readFileName();
writeFile(a);
initialized = true;
}
};

上面readFileName 和initialized = true;沒有必然關係,所以在實際執行的時候,可能會先執行initialized = true;

對於這個線程內的結果沒有影響。

但是如果是多線程的情況下:

    public Runnable mRun2 = new Runnable() {
        @Override
        public void run() {
            while (!initialized)
            {
                try {
                    TraceLog.i("sleep");
                    Thread.sleep(5);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
            doSomeThing(context);
        }
    };

initialized = true的執行順序對線程2的結果有直接的影響。所有有序性在這種情況下,需要保證。

一般java裡面用synchronized就可以保證。但是過多的synchronized會對性能有很大的損失。

4.volatile關鍵字

volatile關鍵字修飾後的變數.有2個作用:

1)用來確保將變數的更新操作通知到其他線程,保證了新值能立即同步到主記憶體,以及每次使用前立即從主記憶體刷新.

當把變數聲明為volatile類型後,編譯器與運行時都會註意到這個變數是共用的.

但是volatile 不能保證線程是安全的,因為java裡面的運算並非原子操作。2)volatile還有一個特性就是保證指令不重新排序。現在編譯器,為了優化代碼,都會重新排序指令。如果在多個線程裡面,就會有很大的問題。

但是指令重排是JVM在它認為合理的情況下做的,所以很難模擬出這一情況。

    boolean aBoolean = false;
    public Runnable mRun1 = new Runnable() {
        @Override
        public void run() {
            aBoolean = false;
            while (!aBoolean)
            {
                doSomeThing();
            }
        }
    };

    public Runnable mRun2 = new Runnable() {
        @Override
        public void run() {
            aBoolean = true;
        }
    };

只是2個線程的例子,線程2用來關閉線程1.一般情況下,它會運行良好,但是有小概率情況下,會有問題。

aBoolean 在賦值為true的時候,沒有立刻被同步到主記憶體,而這時候線程1的工作記憶體aBoolean 的拷貝是false。

所以會陷入死迴圈。

volatile關鍵字就可以避免這種情況的發生。

1)當aBoolean = true;發生後,線程2會立即把aBoolean 的值更新到主記憶體。

2)線程1在使用到aBoolean 是,會首先到主記憶體重新獲取新的值。然後更新工作記憶體中的值,這個時候 aBoolean就是true了,迴圈退出。

5.volatile 保證原子操作嗎?

volatile不能保證線程是安全的。

package com.joyfulmath.jvmexample.multithread;

import com.joyfulmath.jvmexample.TraceLog;

import java.util.concurrent.CountDownLatch;

/**
 * @author deman.lu
 * @version on 2016-05-26 14:34
 */
public class VolatileTest2 {
    public volatile int inc = 0;
    static CountDownLatch countDownLatch = new CountDownLatch(10);
    public void increase() {
        inc++;
    }

    public static void main() {
        TraceLog.i();
        final VolatileTest2 test = new VolatileTest2();
        for(int i=0;i<200;i++){
            new Thread(){
                @Override
                public void run() {
                    super.run();
                    for(int j=0;j<50;j++)
                        test.increase();

                    countDownLatch.countDown();
//                    TraceLog.i(String.valueOf(Thread.currentThread().getId()));
                }
            }.start();
        }

        try {
            countDownLatch.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println(test.inc);
        TraceLog.i(String.valueOf(test.inc));
    }
}
05-26 14:48:06.060 15209-15209/com.joyfulmath.jvmexample I/System.out: 9950
05-26 14:48:06.061 15209-15209/com.joyfulmath.jvmexample I/VolatileTest2: main: 9950 [at (VolatileTest2.java:41)]

結果並不是10000,原因就是 自增函數不是原子操作,而Volatile只能保證數值是更新到住記憶體,但是,當線程1執行過程中假設inc=5,線程2可能已經獲取了inc的值。

這個時候,線程1,++以後變為6,線程2也是6,而且因為主記憶體的值 & 線程2的值一致,就不會觸發其他線程無效的情況,所以線程3取到的值,還是6.所有這個數值的結果是無法確認的,但是<10000.

But, 我在android23下編譯,發現一直是10000.不清楚原因???

6.volatile的有序性

volatile只能保證部分有序性,比如說:

1 volatile boolean initialized = false;
2         public void run() {
3             context = readFileName();
4             writeFile(context);
5             initialized = true;
6             play();
7             Drawable();
8         }

上面,3,4兩行語句順序是亂序的,6,7也是,但是5 一定在3,4之後運行。 也就是5的執行為止不變,而且,3,4 不能和6,7互換執行順序。這就是volatile有限的有序性。

 

參考:

http://www.cnblogs.com/dolphin0520/p/3920373.html

《深入理解java虛擬機》周志明

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 深入理解Java的介面和抽象類 對於面向對象編程來說,抽象是它的一大特征之一。在Java中,可以通過兩種形式來體現OOP的抽象:介面和抽象類。這兩者有太多相似的地方,又有太多不同的地方。很多人在初學的時候會以為它們可以隨意互換使用,但是實際則不然。今天我們就一起來學習一下Java中的介面和抽象類。下 ...
  • 輸入格式: 輸入分2行,每行分別先給出多項式非零項的個數,再以指數遞降方式輸入一個多項式非零項繫數和指數(絕對值均為不超過1000的整數)。數字間以空格分隔。 輸出格式: 輸出分2行,分別以指數遞降方式輸出乘積多項式以及和多項式非零項的繫數和指數。數字間以空格分隔,但結尾不能有多餘空格。零多項式應輸 ...
  • 實現思路:1、使用java.net.URL對象,綁定網路上某一個網頁的地址2、通過java.net.URL對象的openConnection()方法獲得一個HttpConnection對象3、通過HttpConnection對象的getInputStream()方法獲得該網路文件的輸入流對象Inpu ...
  • 學習了正則之後,打算用java玩一玩,所以就決定用它來實現一個好玩的idea import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; //和網路相關的操作 import ...
  • 這就是我們需要抓捕的網站。 下麵是我們的代碼: package cn.bdqn; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; //和網路相關的操作 impor... ...
  • 要想瞭解Spring的事務,首先要瞭解資料庫事務的基本知識,資料庫併發會產生很多問題,Spring使用ThreadLocal技術來處理這些問題,那麼我們必須瞭解Java的ThreadLocal技術。下麵我們逐一瞭解。 第一回合:資料庫事務的基本知識 什麼是資料庫事務? 一次執行多個SQL語句,全部執 ...
  • 簡單電話簿 請用面向對象方法設計並實現一個簡單電話簿,包括如下功能: 1.能設置並以文件保存若幹姓名、聯繫電話和電郵地址。 2.根據輸入的姓名,在已經保存的文件中查詢相應的聯繫電話和電郵地址,並顯示查詢結果。 3.根據輸入的電話號碼,在已經保存的文件中查詢相應的聯繫人姓名,並顯示查詢結果。 ...
  • ibatis改名為mybatis已經將近一年了,mybatis3也發佈了,但是相關的開發文檔卻很少,查閱了官方的使用指南,也是說的模糊不清,一筆帶過,尤其是註解部分,只是列舉了註解種類,卻沒有對應的例子,因此可能會給某些兄弟使用mybatis註解帶來困惑,我想就我微薄的知識想對mybatis註解的使 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...