精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

怎樣才能在不加鎖的情況下解決多線程問題

jf_78858299 ? 來源:碼農的荒島求生 ? 作者:碼農的荒島求生 ? 2023-03-02 09:31 ? 次閱讀

我們知道,多線程同時修改共享變量時會出現數據不一致的問題,比如多個線程同時對一個變量加1,假設count的初始值為0:

int count;
void add() { ++count;}

如果只有一個線程調用add函數,那么什么問題都沒有,但如果多個線程同時調用上述函數,比如10個線程都調用一遍,那么count值最后不一定等于0,原因在于對 count加1的操作不是原子的

所謂某個操作是原子的是指CPU要么執行該操作,要么不執行該操作,不存在中間狀態,但上述對count加1的操作經過編譯器處理后會生成幾條對應的機器指令,所以該操作不是原子的。

那么怎樣才能讓其變成原子的呢?很簡單,加一把鎖。

int count;mutex mtx; // 鎖
void add() { mtx.lock(); ++count; mtx.unlock();};

現在我們用一把鎖將對count的操作保護了起來,此時你可以將mtx.lock()以及mtx.unlock()中間的代碼看成原子的,CPU要完全執行完對count的加1要么根本不會操作count,這樣上述程序的運行結果就是我們想要的了。

這是怎樣做到呢?這就要說到操作系統了,千萬不要小瞧了上面的mutex這把鎖,這把鎖的背后相當復雜,因為這涉及到了操作系統。

假設現在有三個線程,各自運行在不同的CPU核心上,每個方框代表一個時間片:

圖片

T1時間片這三個線程都在調用add函數,線程A拿到鎖,A可以繼續向前推進,但B和C就沒這么幸運了,此時操作系統將剝奪線程B和C繼續持有CPU的權利,將其分配給其它具備執行條件的線程,這就是操作系統中所謂的掛起,注意,這個過程相當復雜,因為這涉及到用戶態與內核態的切換以及線程的切換等等。

此時來到T2時間片,線程A繼續向前推進,線程B和C則被按下暫停鍵。

T3時間片,然而就在線程A拿到鎖運行時因為某些原因像高優先級線程槍占之類導致操作系統也剝奪了線程A繼續持有CPU的權利,糟糕的是,因為線程A此時持有鎖,而線程A又無法繼續向前推進,這就進一步使得線程B和C也無法繼續向前推進。

你會發現在T3時刻,這幾個線程都沒有任何進展,根本原因在于我們為解決多線程問題加互斥鎖驚動了操作系統,而這類互斥鎖是操作系統給我們實現的,那么解決線程安全問題一定要經過操作系統嗎?

不是的,在硬件層面也可以解決線程安全問題,硬件層面當然是指CPU,或者說機器指令。

CPU中有特定的原子指令,實際上操作系統也是基于這些指令實現的互斥鎖,既然操作系統能用這些指令,我們(用戶態)也可以使用這些指令,基于此我們可以將上述代碼進行簡單改造:

int count = 0;void add() {    int old_value;
do { old_value = count; } while (!atomic_compare_exchange(&count, &old_value, old_value + 1));}

此時add函數是線程安全的,我們也沒有對其進行加鎖,不管多少線程同時調用add函數得到count都是正確的, 而該函數的執行完全不涉及操作系統 ,不需要操作系統來維護秩序,利用的就是CPU中的原子指令,CPU在硬件層面一樣可以替我們維護秩序。

上述代碼就是所謂lock-free的, 不管操作系統怎樣調度這三個線程,我們都能確保這三個線程中總有一個能繼續向前推進

lock-free的系統看起來像這樣:

圖片

對于這類系統 不存在某個時間片下線程都無法推進的情況 ,換句話說就是lock-free程序保證至少有一個線程能繼續向前推進。

可以看到,lock-free給出了比普通鎖更優的保障。

但不能簡單從代碼是不是加鎖或不加鎖去判斷代碼是否lock-free ,回旋鎖也是沒有上述互斥鎖的,也不經過操作系統,但回旋鎖并不是lock-free的,如果你這樣利用CPU中的原子操作修改add函數:

int count = 0;int lock = 0;  // 回旋鎖
void add () { int expected = 0; while(!atomic_compare_exchange_weak(&lock, &expected, 1)) expected = 0; count++; lock = 0;}

這就是典型的回旋鎖,然而如果某個線程持有回旋鎖后被操作系統掛起那么其它線程開始無效的執行死循環,除了白白消耗CPU之外它們都無法繼續向前推進,顯而易見,如果此時系統負載較高那么此類程序的性能會變差。

既然現在你已經知道了lock-free我們再繼續優化這段代碼:

std::atomic<int> count;
void add() { ++count;}

這段代碼沒有鎖,也不需要用循環不斷檢測是否有其它線程修改count,不管操作系統如何調度這三個線程, 它們都能在有限的操作數內執行完成 ,此時我們說該程序是wati-free的,wait-free系統運行起來像這樣:

圖片

可以看到在任意時間片內, 不管操作系統怎樣調度,所有線程都能向前推進

wait-free比lock-free的要求更高更加嚴格,由于wait-free的程序總是能在有限的步驟內執行完成,因此實時性是最好的,適用于那些對實時性要求較高的場景,當然實現難度也要比lock-free更高。

值得注意的是,wait-free以及lock-free程序的實現通常不是那么簡單。

好啦,今天就到這里,希望這篇對大家理解多線程有所幫助

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • cpu
    cpu
    +關注

    關注

    68

    文章

    10827

    瀏覽量

    211175
  • 數據
    +關注

    關注

    8

    文章

    6899

    瀏覽量

    88844
  • 多線程
    +關注

    關注

    0

    文章

    277

    瀏覽量

    19923
  • 函數
    +關注

    關注

    3

    文章

    4308

    瀏覽量

    62441
收藏 人收藏

    評論

    相關推薦

    ESP32會不會有多線程問題,需要加鎖嗎?

    ESP32會不會有多線程問題,需要加鎖
    發表于 07-19 08:05

    在不改變內部程序的情況下,只想改變外部元件,如晶振等,怎樣才能讓實時時鐘加快,

    在不改變內部程序的情況下,只想改變外部元件,如晶振等,怎樣才能讓實時時鐘加快,如用1秒的時間跳30秒?
    發表于 05-16 17:55

    labview問題怎樣才能在注冊時信息沒填完整時注冊,

    用簇制作了一個注冊頁面包括密碼,,電話,,姓名,,郵箱等,,但是怎樣才能在注冊時信息沒填完整時注冊,,提醒信息未填寫完整,,請填寫完整。。。
    發表于 07-24 16:04

    Linux多線程機制

    1 線程不能獨立運行,要依附于進程2 如果創建一個子線程只需要重新分配棧空間3 多個線程可以并行運行4 線程之間可以有共同的全局變量(全局區,任何
    發表于 11-11 09:53

    怎樣才能在SDCARD中運行Android系統呢

    怎樣才能在SDCARD中運行Android系統呢?怎樣使用rknand.ko在Android上MBR中定義的mount分區?
    發表于 02-18 06:43

    在MCU開發中使用多線程操作一寫一讀是否需要保護?

    ,那么多線程訪問是安全的,那么對于一寫一讀,在某些情況下需要保護,某些情況下其實可以不需要保護。當操作數據是 1字節 uint8_t 類型數據,可以不做保護,對于uint8_t類型的數據,最終轉化為匯編
    發表于 02-01 15:42

    怎樣才能使本本達到最優性能

    怎樣才能使本本達到最優性能 問題:我是一個最近購本的菜鳥,請問怎樣才能使本本達到最優的性能? 回
    發表于 01-25 14:39 ?519次閱讀

    MFC多線程編程

    計算機上的上位機制作工具語言之MFC多線程編程
    發表于 09-01 14:55 ?0次下載

    Linux多線程編程

    線程呢?使用多線程到底有哪些好處?什么的系統應該選用多線程?我們首先必須回答這些問題。  使用多線程的理由之一是和進程相比,它是一種非常"節儉"的多任務操作方式。我們知道,在Linux
    發表于 04-02 14:43 ?596次閱讀

    在哪幾種情況下會造成伺服電機抖動

    在哪幾種情況下會造成伺服電機抖動?怎樣才能解決這些伺服電機抖動帶來的問題?分別是怎么解決的?
    的頭像 發表于 02-22 16:14 ?1975次閱讀

    在哪幾種情況下會造成伺服電機抖動?

    在哪幾種情況下會造成伺服電機抖動?怎樣才能解決這些伺服電機抖動帶來的問題?分別是怎么解決的?
    發表于 05-24 09:41 ?369次閱讀

    多線程情況下如何對一個值進行 a++ 操作

    多線程情況下,對一個值進行 a++ 操作,會出現什么問題? a++ 的問題 先寫個 demo 的例子。把 a++ 放入多線程中運行一。定義 10 個
    的頭像 發表于 10-13 11:17 ?658次閱讀
    在<b class='flag-5'>多線程</b>的<b class='flag-5'>情況下</b>如何對一個值進行 a++ 操作

    什么情況下避免使用系統調用

    制。如果對變量的每次訪問都使用上述機制,由于系統調用會陷入內核空間,需要頻繁的進行上下文切換,這就導致了程序的時間開銷比較大。 自然的,我們就想到,在多線程環境中,在某些情況下是否能減少甚至避免使用系統調用?答案是肯
    的頭像 發表于 11-13 10:32 ?419次閱讀
    什么<b class='flag-5'>情況下</b>避免使用系統調用

    怎樣才能在有限的容量下發揮電池的極限續航能力

    電子發燒友網站提供《怎樣才能在有限的容量下發揮電池的極限續航能力.doc》資料免費下載
    發表于 11-14 14:38 ?0次下載
    <b class='flag-5'>怎樣才能在</b>有限的容量下發揮電池的極限續航能力

    多線程同步的幾種方法

    多線程同步是指在多個線程并發執行的情況下,為了保證線程執行的正確性和一致性,需要采用特定的方法來協調線程之間的執行順序和共享資源的訪問。下面
    的頭像 發表于 11-17 14:16 ?1128次閱讀