亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

java內存模型與volatile關鍵字介紹

發布時間:2021-09-01 09:48:39 來源:億速云 閱讀:108 作者:chen 欄目:編程語言

這篇文章主要講解了“java內存模型與volatile關鍵字介紹”,文中的講解內容簡單清晰,易于學習與理解,下面請大家跟著小編的思路慢慢深入,一起來研究和學習“java內存模型與volatile關鍵字介紹”吧!

java內存模型可以大致理解分為兩個模塊,主內存和私有內存。主內存中主要是存放一些共享的全局變量,私有內存主要是存放線程所需的私有變量。一般情況下,如果某個線程需要使用主內存的全局變量。首先,它會拷貝一份主內存里面的全局變量到私有內存,進行操作,操作完成以后再把這個變量同步到主內存。如下圖:
java內存模型與volatile關鍵字介紹
如果是單線程的,到沒什么問題,但是如果是多線程的,就有可能出現數據不一致的問題,因為線程之間是不可見的。看下面一個例子:

package org.hzg.volatilekeyword;
/**
 * Created by hzgal on 2019-3-21.
 */
class ThreadVolatileDemo extends Thread {
    public boolean flag = true;
    @Override
    public void run() {
        System.out.println("ThreadVolatileDemo線程開始執行....");
        while (flag) {
        }
        System.out.println("ThreadVolatileDemo線程停止");
    }
    public void stopThread() {
        this.flag = false;
    }
}

public class Demo1 {
    public static void main(String[] args) throws InterruptedException {
        ThreadVolatileDemo threadVolatileDemo = new ThreadVolatileDemo();
        threadVolatileDemo.start();
        Thread.sleep(3000);
        threadVolatileDemo.stopThread();
        System.out.println("flag 已經設置成false");
        Thread.sleep(1000);
        System.out.println(threadVolatileDemo.flag);
    }
}

上面這段程序執行后會發生一個現象,就是會一直執行。原因就在于主線程中雖然對flag進行了修改,但是線程掛起了一秒,導致在主線程私有內存里面的flag沒有同步到主內存中,所以子線程的flag仍然為true,導致自線程一直在執行。解決方案,利用volatile關鍵字,volatile關鍵字有如下含義:
1)保證了不同線程對這個變量進行操作時的可見性,即一個線程修改了某個變量的值,這新值對其他線程來說是立即可見的。
2)禁止進行指令重排序。
也就是說在并發編程中,volatile關鍵字可以在一定程度上保證可見性和有序性。上面的問題這樣處理就可以了。

public volatile boolean flag = true;

但是問題來了,使用volatile有一個弊端,就是它無法保證并發編程下的一致性。看下面這個例子:

package org.hzg.volatilekeyword;
import java.util.ArrayList;
import java.util.List;
/**
 * Created by hzgal on 2019-3-22.
 */
class VolatileDemo2 extends Thread{
    private volatile static int count;
    @Override
    public void run() {
        for (int i = 0; i < 1000; i++) {
            count++;
        }
        System.out.println("線程" + currentThread().getName() + "執行結束,結果為:" + count);
    }
}
public class Demo2 {
    public static void main(String[] args) {
        List<VolatileDemo2> thradList = new ArrayList<VolatileDemo2>(10);
        for (int i = 0; i < 10; i++) {
            thradList.add(new VolatileDemo2());
        }
        for (VolatileDemo2 volatileDemo2 : thradList) {
            volatileDemo2.start();
        }
    }
}

上面的例子會出現如下情況:
java內存模型與volatile關鍵字介紹

這就是volatile關鍵字雖然使得每個線程都對count可見,但是無法保證這些線程對count的操作是原子的。有可能兩個線程同時執行了count++的操作,但是只有一次同步到主內存了。解決上面的問題的方法有很多,其核心的思想就是加鎖,保證對count變量的操作的原子性即可。下面舉兩個例子:
1、利用java并發包里面的工具類。如下:

package org.hzg.volatilekeyword;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.atomic.AtomicInteger;
/**
 * Created by hzgal on 2019-3-22.
 */
class VolatileDemo3 extends Thread{
    private volatile static AtomicInteger count = new AtomicInteger(0);
    @Override
    public void run() {
        for (int i = 0; i < 1000; i++) {
            count.incrementAndGet();
        }
        System.out.println("線程" + currentThread().getName() + "執行結束,結果為:" + count.get());
    }
}
public class Demo3 {
    public static void main(String[] args) {
        List<VolatileDemo3> thradList = new ArrayList<VolatileDemo3>(10);
        for (int i = 0; i < 10; i++) {
            thradList.add(new VolatileDemo3());
        }
        for (VolatileDemo3 volatileDemo3 : thradList) {
            volatileDemo3.start();
        }
    }
}

結果如下:
java內存模型與volatile關鍵字介紹
2、利用synchronized關鍵字

package org.hzg.volatilekeyword;
import java.util.ArrayList;
import java.util.List;
/**
 * Created by hzgal on 2019-3-22.
 */
class VolatileDemo4 extends Thread{
    private volatile static int count;
    private static final String lock = "countLock";
    @Override
    public void run() {
        synchronized (lock) {
            for (int i = 0; i < 1000; i++) {
                count++;
            }
            System.out.println("線程" + currentThread().getName() + "執行結束,結果為:" + count);
        }
    }
}
public class Demo4 {
    public static void main(String[] args) {
        List<VolatileDemo4> thradList = new ArrayList<VolatileDemo4>(10);
        for (int i = 0; i < 10; i++) {
            thradList.add(new VolatileDemo4());
        }
        for (VolatileDemo4 volatileDemo4 : thradList) {
            volatileDemo4.start();
        }
    }
}

結果如下:
java內存模型與volatile關鍵字介紹

感謝各位的閱讀,以上就是“java內存模型與volatile關鍵字介紹”的內容了,經過本文的學習后,相信大家對java內存模型與volatile關鍵字介紹這一問題有了更深刻的體會,具體使用情況還需要大家實踐驗證。這里是億速云,小編將為大家推送更多相關知識點的文章,歡迎關注!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

柏乡县| 潞城市| 赞皇县| 温州市| 嵊泗县| 河曲县| 巫山县| 河南省| 高清| 青州市| 安新县| 兴山县| 家居| 奈曼旗| 平原县| 晴隆县| 习水县| 陇南市| 鄂伦春自治旗| 丹巴县| 游戏| 万荣县| 太仓市| 尤溪县| 岢岚县| 建水县| 平谷区| 满城县| 泸定县| 石狮市| 宁海县| 化隆| 鹤岗市| 云和县| 定远县| 西畴县| 祁门县| 大同市| 乌拉特后旗| 隆昌县| 遂昌县|