主頁 > 後端開發 > 關于執行緒的快取重繪

關于執行緒的快取重繪

2023-05-23 08:32:03 後端開發

今天又是摸魚的一天,在群里閑聊的時候突然有位群友題了個問題:

群友們反應很快,一下子就解決了沒有加關鍵字volatile,不能保證可見性,所以回圈執行緒中的kaiguan變數一直讀取的是執行緒快取中的值,而執行緒快取的值一直無法讀取到主記憶體中的值,

當然這個問題難不倒這位提問的群友,所以他又提問了,

除了加volatile之外,還有沒有其他方式?

正好閑著無聊,我便開始試著解決這個問題,

public class test2 {

    private static boolean kaiguan =true;
    private static void go(){
        new Thread(()->{
            while (kaiguan){
            }

            System.out.println("事情做完了");
        }).start();
        new Thread(()->{
            try {
                Thread.sleep(1000*3);
                kaiguan=false;
            }catch (InterruptedException e){
                e.printStackTrace();
            }
        }).start();
    }

    public static void main(String[] args) {
        go();
    }


}

首先我問了ChatGPT

如果不使用 volatile 關鍵字,也可以使用其他的多執行緒機制來保證可見性,例如使用 AtomicBoolean 類,這個類提供了 get()set() 方法,可以保證操作的原子性和可見性,
修改后的代碼如下所示:
private static AtomicBoolean kaiguan = new AtomicBoolean(true);
private static void go() {
new Thread(() -> {
while (kaiguan.get()) {
Thread.yield(); // 或者使用Thread.sleep(),暫停執行緒一段時間
}
System.out.println("事情做完了");
}).start();
new Thread(() -> {
try {
Thread.sleep(1000 * 3);
kaiguan.set(false); // 設定kaiguan變數為false,退出回圈
} catch (InterruptedException e) {
e.printStackTrace();
}
}).start();
}
public static void main(String[] args) {
go();
}
在上述代碼中,我們使用了 AtomicBoolean 類來代替原來的 boolean 型別的 kaiguan 變數,并在 while 回圈中使用了 kaiguan.get() 方法來獲取 kaiguan 變數的值,從而保證可見性,在另一個執行緒中,我們使用 kaiguan.set(false) 方法將 kaiguan 變數設定為 false ,從而退出回圈,

可以看到 除了使用volatile 來宣告變數,我們還可以用AtomicBoolean 來宣告kaiguan,同樣能保證可見性,

那如果不用AtomicBoolean宣告kaiguan呢?

我看到了下面這篇文章:

執行緒的快取何時重繪? - 簡書 (jianshu.com)

在Doug Lea大神的Concurrent Programming in Java一書中有這樣一個片段來描述synchronized這個關鍵字:

In essence, releasing a lock forces a flush of all writes from working memory employed by the thread, and acquiring a lock forces a (re)load of the values of accessible fields. While lock actions provide exclusion only for the operations performed within a synchronized method or block, these memory effects are defined to cover all fields used by the thread performing the action.

簡單翻譯一下:從本質上來說,當執行緒釋放一個鎖時會強制性的將作業記憶體中之前所有的寫操作都重繪到主記憶體中去,而獲取一個鎖則會強制性的加載可訪問到的值到執行緒作業記憶體中來,雖然鎖操作只對同步方法和同步代碼塊這一塊起到作用,但是影響的卻是執行緒執行操作所使用的所有欄位,

也就是說我們可以用加鎖來解決執行緒重繪這個問題,

所以我們可以手動加上System.out.println();來退出該回圈,

因為System.out.println();底層是加鎖的

    public void println() {
        newLine();
    }
    
        private void newLine() {
        try {
            synchronized (this) {
                ensureOpen();
                textOut.newLine();
                textOut.flushBuffer();
                charOut.flushBuffer();
                if (autoFlush)
                    out.flush();
            }
        }
        catch (InterruptedIOException x) {
            Thread.currentThread().interrupt();
        }
        catch (IOException x) {
            trouble = true;
        }
    }
    

我們在看看chatgpt的回答,既然里面有提到yield()與sleep(),那么我們就試試

Thread.yield();
與
Thread.sleep(0);

發現果然成功跳出回圈,那么yield()與sleep(0)到底發生了什么導致快取重繪呢?

沒錯就是背景關系切換!

yield()與sleep(0)會導致背景關系切換,從而導致快取失效,從而拉去主記憶體中的新值,

當然我們也可以直接使用Unsafe方法中的loadFence()方法,

使用UnsafeFactory.getUnsafe().loadFence();也同樣可以跳出回圈,因為loadFence: 可以保證在這個屏障之前的所有讀操作都已經完成,

Unsafe需要我們通過反射獲取,直接呼叫會報錯:

  public static Unsafe getUnsafe() {
        try {
            Field field = Unsafe.class.getDeclaredField("theUnsafe");
            field.setAccessible(true);
            return (Unsafe) field.get(null);
        } catch (Exception e) {
            e.printStackTrace();
        }
        return null;
    }

在尋找其他答案的程序中我還發現final關鍵字同樣也有重繪快取的作用

首先自定義一個類,將其中的num設定為final


public  class TestTempOne {
  final int num; //設為final;
    TestTempOne(int num){
        this.num=num;
    }

}

private static void go(){
        new Thread(()->{
            while (kaiguan){
//                new TestTempOne(0);  //跳出回圈
//               new String("a");      //跳出回圈
//                new Integer(0);      //死回圈
//                new Integer(129); //死回圈
                new Integer(100000000); //死回圈
            }

可以看到最侄訓跳出回圈,但是有個問題Integer中value的值同樣也是final,但卻不能重繪快取,而String則是一個final char陣列,也可以跳出回圈,目前沒有找到答案,如果有大佬知道答案,請告知我一下!

總結:我目前知道的有六種:

1、使用volatile

2、使用synchronized或者Lock

3、使用AtomicBoolean

4、使用UnsafeFactory.getUnsafe().loadFence();

5、使用yield()與sleep()

6、final關鍵字

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/553077.html

標籤:其他

上一篇:Java中的三元運算,以后用得到!

下一篇:返回列表

標籤雲
其他(159463) Python(38162) JavaScript(25441) Java(18096) C(15230) 區塊鏈(8267) C#(7972) AI(7469) 爪哇(7425) MySQL(7204) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5871) 数组(5741) R(5409) Linux(5340) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4574) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2433) ASP.NET(2403) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) .NET技术(1975) 功能(1967) Web開發(1951) HtmlCss(1940) C++(1919) python-3.x(1918) 弹簧靴(1913) xml(1889) PostgreSQL(1878) .NETCore(1861) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • 關于執行緒的快取重繪

    今天又是摸魚的一天,在群里閑聊的時候突然有位群友題了個問題: ![](https://img2023.cnblogs.com/blog/2696704/202305/2696704-20230522233309409-1620806525.png) 群友們反應很快,一下子就解決了沒有加關鍵字vola ......

    uj5u.com 2023-05-23 08:32:03 more
  • Java中的三元運算,以后用得到!

    # 前言 Java 中的三元運算,平時也叫做三目運算,大家了解嗎?下面就詳細介紹一下,以后在專案編程中用得到。 # 一、Java運算子 在最底層,Java 中的資料是通過使用運算子來操作的。運算子是一種特殊的符號,用來表示資料的運算、賦值和比較等等。每一種編程語言都有運算子,在 Java 中運算子可 ......

    uj5u.com 2023-05-23 08:31:57 more
  • Java設計模式-組合模式

    # 簡介 在軟體設計中,設計模式是一種被廣泛接受和應用的經驗總結,旨在解決常見問題并提供可復用的解決方案。 組合模式是一種結構型設計模式,它允許將物件組合成樹形結構以表示“部分-整體”的層次結構。這種模式能夠使客戶端以一致的方式處理單個物件和物件集合,將物件的組合與物件的使用具有一致性。 與其他設計 ......

    uj5u.com 2023-05-23 08:31:53 more
  • Stream流體系

    視頻地址 # 1 Stream流概述 - 目的:簡化集合和陣列操作的API,結合了Lambda運算式。 - Stream流式思想的核心: 1. 先得到集合或者陣列的Stream流(就是一根傳送帶) 2. 把元素放上去 3. 用這個Stream流簡化的API來方便的操作元素 # 2 Stream流獲取 ......

    uj5u.com 2023-05-23 08:31:40 more
  • python呼叫父類方法的三種方式(super呼叫和父類名呼叫)

    ### 子類呼叫父類的方法的三種方式: - 父類名.方法名(self) - super(子類名,self).父類方法名() - super().父類方法名 注意:super()通過子類呼叫當前父類的方法,super默認會呼叫第一個父類的方法(適用于單繼承的多層繼承 如下代碼: ```python # ......

    uj5u.com 2023-05-23 08:31:31 more
  • 用Python將女朋友的照片做成壁紙軟體,實作桌面壁紙自動更換!

    話說兄弟們,女朋友生氣了都是怎么哄的? 不會吧不會吧,不會有人還是單身狗吧! 算了,還是回到正題吧,再說我要挨打了~ 今天咱們來交流一下程式員是怎么哄女朋友的,話不多說直接開始! 準備作業 1、環境 首先我們準備好環境和編輯器,我使用的是: Python 3.8 解釋器 Pycharm 編輯器 2、 ......

    uj5u.com 2023-05-23 08:26:16 more
  • 如何通過Java代碼將 PDF檔案轉為 HTML格式

    雖然PDF檔案適合用于列印和發布,但不適合所有型別的檔案。例如,包含復雜圖表和圖形的檔案可能無法在PDF中呈現得很好。但是HTML檔案可以在任何可運行瀏覽器的計算機上進行閱讀并顯示。并且HTML還具有占用服務器資源較小,便于搜索引擎收錄的特點。那么今天這篇文章就將展示如何通過Java應用程式將PDF ......

    uj5u.com 2023-05-23 08:21:05 more
  • Python豎版大屏 | 用pyecharts開發可視化的奇妙探索!

    你好!我是[@馬哥python說](https://www.zhihu.com/people/13273183132),一枚10年程式猿👨🏻?💻,正在試錯用pyecharts開發可視化大屏的非常規排版。 以下,我用8種ThemeType展示的同一個可視化資料大屏。 **1、SHINE主題** ......

    uj5u.com 2023-05-23 08:15:32 more
  • Java網路編程----通過實作簡易聊天工具來聊聊NIO

    前文我們說過了BIO,今天我們聊聊NIO。NIO 是什么?NIO官方解釋它為New lO,由于其特性我們也稱之為,Non-Blocking IO。這是jdk1.4之后新增的一套IO標準。為什么要用NIO呢?我們再簡單回顧下BIO:阻塞式IO,原理很簡單,其實就是多個端點與服務端進行通信時,每個客戶端 ......

    uj5u.com 2023-05-23 08:10:11 more
  • 論elasticsearch在Windows環境的安裝

    # 前置需求 一臺電腦(我用的是Windows),有網 # 第一步:下載并安裝 去java官網下載開發版java(考慮到可能有小白,我暫且這么說) java官網下載鏈接:https://www.oracle.com/java/technologies/downloads/ 寫隨筆時間為2023、05 ......

    uj5u.com 2023-05-23 08:09:32 more