主頁 >  其他 > Linux_深究多執行緒_(執行緒等待,執行緒終止,執行緒分離,執行緒互斥,可重入,執行緒安全)

Linux_深究多執行緒_(執行緒等待,執行緒終止,執行緒分離,執行緒互斥,可重入,執行緒安全)

2021-04-25 14:17:11 其他

文章目錄

  • 1. 主 / 新執行緒
    • 1.1 主 / 新執行緒退出
  • 2.執行緒等待
  • 3.執行緒終止
  • 4.部分總結
  • 5.執行緒分離
  • 6.執行緒分離總結
  • 7.執行緒互斥
    • 7.1 執行緒間相關概念
    • 7.2 互斥量mutex
    • 7.3 互斥量的介面
    • 7.4 代碼示例
    • 7.5 互斥量實作原理
  • 8. 互斥量總結
  • 9. 可重入 & 執行緒安全
    • 9.1 概念
    • 9.2 常見的執行緒不安全的情況
    • 9.3 常見的執行緒安全的情況
    • 9.4 常見不可重入的情況
    • 9.5 常見可重入的情況
    • 9.6 可重入與執行緒安全聯系
    • 9.7 可重入與執行緒安全區別

  • 在前面的博客【Linux_初始多執行緒—>Link】中已經講了有關多執行緒的概念和基本知識,這章將接著上一章的內容對執行緒操作以及其他概念繼續深究,

1. 主 / 新執行緒

在這里插入圖片描述

  • 從上圖中可得知main()函式中,函式pthread_create()往下的為主執行緒,而函式pthread_create()創造的執行緒為新執行緒,接下來的內容將圍繞主/新執行緒講解,
  • 注意執行緒之間誰先執行是不確定的,而跟系統調度有關,(執行緒是調度的基本單位,系統調度的執行緒標識是LWP)

1.1 主 / 新執行緒退出

  • 主執行緒或者新執行緒各自獨立退出時會有兩種情況
  1. 主執行緒沒有等待新執行緒結束而提前退出,則整個行程退出
    問題:沒有等待新執行緒退出而行程退出,新執行緒也是該行程中的一個執行流,會造成記憶體泄漏,

在這里插入圖片描述

  1. 新執行緒退出而主執行緒不退出,主執行緒繼續執行

在這里插入圖片描述

2.執行緒等待

  • 為什么需要執行緒等待?
  1. 已經退出的執行緒,其空間沒有被釋放,仍然在行程的地址空間內,(如上面的情況1
  2. 創建新的執行緒不會復用剛才退出執行緒的地址空間,
int pthread_join(pthread_t thread, void **value_ptr);

引數

  • thread:執行緒ID,
  • value_ptr:它指向一個指標,后者指向執行緒的回傳值,

回傳值:成功回傳0;失敗回傳錯誤碼,

呼叫該函式的執行緒將掛起等待,直到id為thread的執行緒終止,
在這里插入圖片描述

3.執行緒終止

  • 如果需要只終止某個執行緒而不終止整個行程,可以有三種方法:
  1. 從執行緒函式return,這種方法對主執行緒不適用,從main函式return相當于呼叫exit
  2. 執行緒可以呼叫pthread_ exit終止自己
  3. 一個執行緒可以呼叫pthread_ cancel終止同一行程中的另一個執行緒
  • 從執行緒函式return,這種方法對主執行緒不適用,從main函式return相當于呼叫exit,在main()函式中return則是結束的整個函式,如果在呼叫的函式中return則只是結束這個函式,而exit()是不管在那個位置,都直接結束整個函式,執行緒也是如此,
    在這里插入圖片描述

  • pthread_exit()函式

void pthread_exit(void *value_ptr)

引數

  • value_ptr:value_ptr不要指向一個區域變數,

回傳值:無回傳值,跟行程一樣,執行緒結束的時候無法回傳到它的呼叫者(自身),

在這里插入圖片描述

  • pthread_ cancel()函式
int pthread_cancel(pthread_t thread);

引數:

  • thread:執行緒ID

回傳值

  • 成功回傳0,失敗回傳錯誤碼,
    在這里插入圖片描述
  • 為什么被pthread_cancel()退出以后退出碼是-1呢

答:用pthreat_cancel()函式退出別的別執行緒以后,該執行緒是被例外終止掉的,value_ ptr所指向的單元里存放的是常數PTHREAD_CANCELED,該函式是一個宏定義,就是-1
在這里插入圖片描述

4.部分總結

  • thread執行緒以不同的方法終止,通過pthread_join得到的終止狀態是不同的,總結如下:
  1. 如果thread執行緒通過return回傳,value_ ptr所指向的單元里存放的是thread執行緒函式的回傳值,
  2. 如果thread執行緒被別的執行緒呼叫pthread_ cancel例外終掉,value_ ptr所指向的單元里存放的是常數PTHREAD_CANCELED
  3. 如果thread執行緒是自己呼叫pthread_exit終止的,value_ptr所指向的單元存放的是傳給pthread_exit的引數,
  4. 如果對thread執行緒的終止狀態不感興趣,可以傳NULL給value_ ptr參

5.執行緒分離

  • 當不需要關心某個執行緒執行結果如何,或者執行的對與錯,就可將這個執行緒進行分離,執行緒分離帶來的最直觀的感受就是主執行緒不需要關心這個執行緒的執行如何,
  1. 默認情況下,新創建的執行緒是joinable的,執行緒退出后,需要對其進行pthread_join操作,否則無法釋放資源,從而造成系統泄漏,
  2. 如果不關心執行緒的回傳值,join是一種負擔,這個時候,我們可以告訴系統,當執行緒退出時,自動釋放執行緒資源,
  3. 可以是執行緒組內其他執行緒對目標執行緒進行分離,也可以是執行緒自己分離:
  • 在別的執行緒中分離
int pthread_detach(pthread_t thread);
  • 自己分離自己
pthread_detach(pthread_self());

在這里插入圖片描述

6.執行緒分離總結

  1. 執行緒分離退出碼是0,而不是執行緒回傳的退出碼,是因為將執行緒分離以后,拿不到執行緒的退出碼,也不需要拿到它的退出碼,
  2. 執行緒分離以后,被分離的執行緒會自動釋放自己的資源,不需要被等待,退出碼也不會寫進PCB,
  3. 但是執行緒被分離以后,如果執行緒中有例外,還是會影響當前行程,(舉例:分離以后,各過各的,但是我出現問題,還是會影響你),

7.執行緒互斥

7.1 執行緒間相關概念

  1. 臨界資源:多執行緒執行流共享的資源就叫做臨界資源,(共享資料)
  2. 臨界區:每個執行緒內部,訪問臨界區域的代碼,就叫做臨界區,(共享代碼)
  3. 互斥:任何時刻,互斥保證有且只有一個執行流進入臨界區,訪問臨界資源,通常對臨界資源起保護作用
  4. 原子性(后面討論如何實作):不會被任何調度機制打斷的操作,該操作只有兩態,要么完成,要么未完成,

在這里插入圖片描述

7.2 互斥量mutex

  1. 大部分情況,執行緒使用的資料都是區域變數,變數的地址空間在執行緒堆疊空間內,這種情況,變數歸屬單個執行緒,其他執行緒無法獲得這種變數,
  2. 但有時候,很多變數都需要在執行緒間共享,這樣的變數稱為共享變數,可以通過資料的共享,完成執行緒之間的互動,
  3. 多個執行緒并發的操作共享變數,會帶來一些問題
    在這里插入圖片描述
  • 要解決以上問題,需要做到三點
  1. 代碼必須要有互斥行為:當代碼進入臨界區執行時,不允許其他執行緒進入該臨界區,
  2. 如果多個執行緒同時要求執行臨界區的代碼,并且臨界區沒有執行緒在執行,那么只能允許一個執行緒進入該臨界區,
  3. 如果執行緒不在臨界區中執行,那么該執行緒不能阻止其他執行緒進入臨界區,

注意:要做到這三點,本質上就是需要一把鎖,Linux上提供的這把鎖叫互斥量

7.3 互斥量的介面

初始化互斥量

  • 初始化互斥量有兩種方法:
  1. 靜態分配:
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER
  1. 動態分配:
int pthread_mutex_init(pthread_mutex_t *restrict mutex, const pthread_mutexattr_t
*restrict attr);

引數

  • mutex:要初始化的互斥量
  • attr:NULL

銷毀互斥量

int pthread_mutex_destroy(pthread_mutex_t *mutex)

銷毀互斥量需要注意

  1. 使用PTHREAD_ MUTEX_ INITIALIZER初始化的互斥量不需要銷毀
  2. 不要銷毀一個已經加鎖的互斥量
  3. 已經銷毀的互斥量,要確保后面不會有執行緒再嘗試加鎖

互斥量加鎖和解鎖

int pthread_mutex_lock(pthread_mutex_t *mutex);
int pthread_mutex_unlock(pthread_mutex_t *mutex);

回傳值:

  • 成功回傳0,失敗回傳錯誤號,

呼叫pthread_ lock 時,可能會遇到以下情況:

  1. 互斥量處于未鎖狀態,該函式會將互斥量鎖定,同時回傳成功,
  2. 發起函式呼叫時,其他執行緒已經鎖定互斥量,或者存在其他執行緒同時申請互斥量,但沒有競爭到互斥量,那么pthread_ lock呼叫會陷入阻塞(執行流被掛起),等待互斥量解鎖,

7.4 代碼示例

在這里插入圖片描述

7.5 互斥量實作原理

  1. 經過上面的例子,如果不是原子的,有可能會有資料出現問題,
  2. 為了實作互斥鎖操作,大多數體系結構都提供了swap或exchange指令,該指令的作用是把暫存器和記憶體單元的資料相交換,由于只有一條指令,保證了原子性,即使是多處理器平臺,訪問記憶體的 總線周期也有先后,一個處理器上的交換指令執行時另一個處理器的交換指令只能等待總線周期, 鎖的原子性詳解->link,

8. 互斥量總結

  1. 對臨界區進行保護,所有的執行執行緒都必須遵守這個規則,
  2. 基本程序:lock->訪問臨界區->unlock,鎖的原子性詳解->link,
  3. 所有的執行緒必須先看到同一把鎖,鎖本身就是臨界資源!鎖本身得先保證自身安全!申請鎖的程序,不能有中間狀態,也就是兩態的,lock->原子性,unlock->原子性,
  4. lock->訪問臨界區(花時間)->unlock,在特定執行緒/行程擁有鎖的時候,期間有新執行緒過來申請鎖,一定不能申請到,那么新執行緒進行阻塞,將行程,執行緒對應的PCB投入到等待佇列,
  5. 加鎖時候效率變低,本來是并行/并發執行流,加鎖以后成了串行執行流
  6. 一次保證只有一個執行緒進入臨界區,訪問臨界資源,就叫做互斥

9. 可重入 & 執行緒安全

9.1 概念

  • 執行緒安全:多個執行緒并發同一段代碼時,不會出現不同的結果,常見對全域變數或者靜態變數進行操作,
    并且沒有鎖保護的情況下,會出現該問題,
  • 重入:同一個函式被不同的執行流呼叫,當前一個流程還沒有執行完,就有其他的執行流再次進入,我們
    稱之為重入,一個函式在重入的情況下,運行結果不會出現任何不同或者任何問題,則該函式被稱為可重
    入函式,否則,是不可重入函式

9.2 常見的執行緒不安全的情況

  1. 不保護共享變數的函式
  2. 函式狀態隨著被呼叫,狀態發生變化的函式
  3. 回傳指向靜態變數指標的函式
  4. 呼叫執行緒不安全函式的函式

9.3 常見的執行緒安全的情況

  1. 每個執行緒對全域變數或者靜態變數只有讀取的權限,而沒有寫入的權限,一般來說這些執行緒是安全的
  2. 類或者介面對于執行緒來說都是原子操作
  3. 多個執行緒之間的切換不會導致該介面的執行結果存在二義性

9.4 常見不可重入的情況

  1. 呼叫了malloc/free函式,因為malloc函式是用全域鏈表來管理堆的
  2. 呼叫了標準I/O庫函式,標準I/O庫的很多實作都以不可重入的方式使用全域資料結構
  3. 可重入函式體內使用了靜態的資料結構

9.5 常見可重入的情況

  1. 不使用全域變數或靜態變數
  2. 不使用用malloc或者new開辟出的空間
  3. 不呼叫不可重入函式
  4. 不回傳靜態或全域資料,所有資料都有函式的呼叫者提供
  5. 使用本地資料,或者通過制作全域資料的本地拷貝來保護全域資料

9.6 可重入與執行緒安全聯系

  1. 函式是可重入的,那就是執行緒安全的
  2. 函式是不可重入的,那就不能由多個執行緒使用,有可能引發執行緒安全問題
  3. 如果一個函式中有全域變數,那么這個函式既不是執行緒安全也不是可重入的

9.7 可重入與執行緒安全區別

  1. 可重入函式是執行緒安全函式的一種
  2. 執行緒安全不一定是可重入的,而可重入函式則一定是執行緒安全的,
  3. 如果將對臨界資源的訪問加上鎖,則這個函式是執行緒安全的,但如果這個重入函式若鎖還未釋放則會產生死鎖,因此是不可重入的

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

標籤:其他

上一篇:GUI輸入框監聽事件

下一篇:本人停更告知

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

熱門瀏覽
  • 網閘典型架構簡述

    網閘架構一般分為兩種:三主機的三系統架構網閘和雙主機的2+1架構網閘。 三主機架構分別為內端機、外端機和仲裁機。三機無論從軟體和硬體上均各自獨立。首先從硬體上來看,三機都用各自獨立的主板、記憶體及存盤設備。從軟體上來看,三機有各自獨立的作業系統。這樣能達到完全的三機獨立。對于“2+1”系統,“2”分為 ......

    uj5u.com 2020-09-10 02:00:44 more
  • 如何從xshell上傳檔案到centos linux虛擬機里

    如何從xshell上傳檔案到centos linux虛擬機里及:虛擬機CentOs下執行 yum -y install lrzsz命令,出現錯誤:鏡像無法找到軟體包 前言 一、安裝lrzsz步驟 二、上傳檔案 三、遇到的問題及解決方案 總結 前言 提示:其實很簡單,往虛擬機上安裝一個上傳檔案的工具 ......

    uj5u.com 2020-09-10 02:00:47 more
  • 一、SQLMAP入門

    一、SQLMAP入門 1、判斷是否存在注入 sqlmap.py -u 網址/id=1 id=1不可缺少。當注入點后面的引數大于兩個時。需要加雙引號, sqlmap.py -u "網址/id=1&uid=1" 2、判斷文本中的請求是否存在注入 從文本中加載http請求,SQLMAP可以從一個文本檔案中 ......

    uj5u.com 2020-09-10 02:00:50 more
  • Metasploit 簡單使用教程

    metasploit 簡單使用教程 浩先生, 2020-08-28 16:18:25 分類專欄: kail 網路安全 linux 文章標簽: linux資訊安全 編輯 著作權 metasploit 使用教程 前言 一、Metasploit是什么? 二、準備作業 三、具體步驟 前言 Msfconsole ......

    uj5u.com 2020-09-10 02:00:53 more
  • 游戲逆向之驅動層與用戶層通訊

    驅動層代碼: #pragma once #include <ntifs.h> #define add_code CTL_CODE(FILE_DEVICE_UNKNOWN,0x800,METHOD_BUFFERED,FILE_ANY_ACCESS) /* 更多游戲逆向視頻www.yxfzedu.com ......

    uj5u.com 2020-09-10 02:00:56 more
  • 北斗電力時鐘(北斗授時服務器)讓網路資料更精準

    北斗電力時鐘(北斗授時服務器)讓網路資料更精準 北斗電力時鐘(北斗授時服務器)讓網路資料更精準 京準電子科技官微——ahjzsz 近幾年,資訊技術的得了快速發展,互聯網在逐漸普及,其在人們生活和生產中都得到了廣泛應用,并且取得了不錯的應用效果。計算機網路資訊在電力系統中的應用,一方面使電力系統的運行 ......

    uj5u.com 2020-09-10 02:01:03 more
  • 【CTF】CTFHub 技能樹 彩蛋 writeup

    ?碎碎念 CTFHub:https://www.ctfhub.com/ 筆者入門CTF時時剛開始刷的是bugku的舊平臺,后來才有了CTFHub。 感覺不論是網頁UI設計,還是題目質量,賽事跟蹤,工具軟體都做得很不錯。 而且因為獨到的金幣制度的確讓人有一種想去刷題賺金幣的感覺。 個人還是非常喜歡這個 ......

    uj5u.com 2020-09-10 02:04:05 more
  • 02windows基礎操作

    我學到了一下幾點 Windows系統目錄結構與滲透的作用 常見Windows的服務詳解 Windows埠詳解 常用的Windows注冊表詳解 hacker DOS命令詳解(net user / type /md /rd/ dir /cd /net use copy、批處理 等) 利用dos命令制作 ......

    uj5u.com 2020-09-10 02:04:18 more
  • 03.Linux基礎操作

    我學到了以下幾點 01Linux系統介紹02系統安裝,密碼啊破解03Linux常用命令04LAMP 01LINUX windows: win03 8 12 16 19 配置不繁瑣 Linux:redhat,centos(紅帽社區版),Ubuntu server,suse unix:金融機構,證券,銀 ......

    uj5u.com 2020-09-10 02:04:30 more
  • 05HTML

    01HTML介紹 02頭部標簽講解03基礎標簽講解04表單標簽講解 HTML前段語言 js1.了解代碼2.根據代碼 懂得挖掘漏洞 (POST注入/XSS漏洞上傳)3.黑帽seo 白帽seo 客戶網站被黑帽植入劫持代碼如何處理4.熟悉html表單 <html><head><title>TDK標題,描述 ......

    uj5u.com 2020-09-10 02:04:36 more
最新发布
  • 2023年最新微信小程式抓包教程

    01 開門見山 隔一個月發一篇文章,不過分。 首先回顧一下《微信系結手機號資料庫被脫庫事件》,我也是第一時間得知了這個訊息,然后跟蹤了整件事情的經過。下面是這起事件的相關截圖以及近日流出的一萬條資料樣本: 個人認為這件事也沒什么,還不如關注一下之前45億快遞資料查詢渠道疑似在近日復活的訊息。 訊息是 ......

    uj5u.com 2023-04-20 08:48:24 more
  • web3 產品介紹:metamask 錢包 使用最多的瀏覽器插件錢包

    Metamask錢包是一種基于區塊鏈技術的數字貨幣錢包,它允許用戶在安全、便捷的環境下管理自己的加密資產。Metamask錢包是以太坊生態系統中最流行的錢包之一,它具有易于使用、安全性高和功能強大等優點。 本文將詳細介紹Metamask錢包的功能和使用方法。 一、 Metamask錢包的功能 數字資 ......

    uj5u.com 2023-04-20 08:47:46 more
  • vulnhub_Earth

    前言 靶機地址->>>vulnhub_Earth 攻擊機ip:192.168.20.121 靶機ip:192.168.20.122 參考文章 https://www.cnblogs.com/Jing-X/archive/2022/04/03/16097695.html https://www.cnb ......

    uj5u.com 2023-04-20 07:46:20 more
  • 從4k到42k,軟體測驗工程師的漲薪史,給我看哭了

    清明節一過,盲猜大家已經無心上班,在數著日子準備過五一,但一想到銀行卡里的余額……瞬間心情就不美麗了。最近,2023年高校畢業生就業調查顯示,本科畢業月平均起薪為5825元。調查一出,便有很多同學表示自己又被平均了。看著這一資料,不免讓人想到前不久中國青年報的一項調查:近六成大學生認為畢業10年內會 ......

    uj5u.com 2023-04-20 07:44:00 more
  • 最新版本 Stable Diffusion 開源 AI 繪畫工具之中文自動提詞篇

    🎈 標簽生成器 由于輸入正向提示詞 prompt 和反向提示詞 negative prompt 都是使用英文,所以對學習母語的我們非常不友好 使用網址:https://tinygeeker.github.io/p/ai-prompt-generator 這個網址是為了讓大家在使用 AI 繪畫的時候 ......

    uj5u.com 2023-04-20 07:43:36 more
  • 漫談前端自動化測驗演進之路及測驗工具分析

    隨著前端技術的不斷發展和應用程式的日益復雜,前端自動化測驗也在不斷演進。隨著 Web 應用程式變得越來越復雜,自動化測驗的需求也越來越高。如今,自動化測驗已經成為 Web 應用程式開發程序中不可或缺的一部分,它們可以幫助開發人員更快地發現和修復錯誤,提高應用程式的性能和可靠性。 ......

    uj5u.com 2023-04-20 07:43:16 more
  • CANN開發實踐:4個DVPP記憶體問題的典型案例解讀

    摘要:由于DVPP媒體資料處理功能對存放輸入、輸出資料的記憶體有更高的要求(例如,記憶體首地址128位元組對齊),因此需呼叫專用的記憶體申請介面,那么本期就分享幾個關于DVPP記憶體問題的典型案例,并給出原因分析及解決方法。 本文分享自華為云社區《FAQ_DVPP記憶體問題案例》,作者:昇騰CANN。 DVPP ......

    uj5u.com 2023-04-20 07:43:03 more
  • msf學習

    msf學習 以kali自帶的msf為例 一、msf核心模塊與功能 msf模塊都放在/usr/share/metasploit-framework/modules目錄下 1、auxiliary 輔助模塊,輔助滲透(埠掃描、登錄密碼爆破、漏洞驗證等) 2、encoders 編碼器模塊,主要包含各種編碼 ......

    uj5u.com 2023-04-20 07:42:59 more
  • Halcon軟體安裝與界面簡介

    1. 下載Halcon17版本到到本地 2. 雙擊安裝包后 3. 步驟如下 1.2 Halcon軟體安裝 界面分為四大塊 1. Halcon的五個助手 1) 影像采集助手:與相機連接,設定相機引數,采集影像 2) 標定助手:九點標定或是其它的標定,生成標定檔案及內參外參,可以將像素單位轉換為長度單位 ......

    uj5u.com 2023-04-20 07:42:17 more
  • 在MacOS下使用Unity3D開發游戲

    第一次發博客,先發一下我的游戲開發環境吧。 去年2月份買了一臺MacBookPro2021 M1pro(以下簡稱mbp),這一年來一直在用mbp開發游戲。我大致分享一下我的開發工具以及使用體驗。 1、Unity 官網鏈接: https://unity.cn/releases 我一般使用的Apple ......

    uj5u.com 2023-04-20 07:40:19 more