Linux:線程的概念

在這里插入圖片描述

個人主頁 : 個人主頁
個人專欄 : 《數據結構》 《C語言》《C++》《Linux》

文章目錄

  • 前言
  • 一、線程的概念
    • 線程代碼的簡單示例
  • 總結


前言

本文是對于線程概念的知識總結


一、線程的概念

在課本上,線程是比進程更輕量級的一種指向流 或 線程是在進程內部執行的一種執行流。
我們再提出兩個理解,線程是CPU調度的基本單位 / 進程是承擔系統資源的基本實體。
先記住上面的結論
我們知道,進程 = 內核數據結構 + 代碼和數據構成的。
在這里插入圖片描述
CPU要調度進程,就要有運行隊列,而運行隊列中排隊的就是pcb。CPU通過這些pcb,找到對應的地址空間,進而通過地址空間中的虛擬地址,在頁表中映射物理地址,從而找到對應的代碼和數據。那么,我們是不是可以將地址空間理解為進程的資源窗口,畢竟進程想要訪問正文代碼,數據,new和malloc的空間,共享庫,棧上的臨時數據,命令行參數和環境變量等都是通過地址空間來進行的。

那么,我們如果要創建進程,就要創建對應的pcb,地址空間,將磁盤中的代碼和數據加載進內存,再將地址空間中的虛擬地址與物理地址映射構成頁表,打開stdin,stdout,stderr構建文件資源描述表,初始化信號處理過程等,這樣看來進程創建的成本還挺高的。那為了減少成本,我們能不能在進程內部,再創建多個pcb指向該進程的地址空間,將代碼分成多個,并將私有的數據,使每個pcb各自私有一份,可以共享的數據就共享。當CPU來調度其中一個pcb時,其只會運行該進程的一部分代碼和一部分數據。我們就可以將這種比以往進程更輕(創建成本)的東西,稱為線程。

在這里插入圖片描述
在linux程序員看來,描述線程的結構體(TCB Thread control block ) 中屬性在pcb中都有。那如果我們把pcb來充當tcb,我們就可以把進程調度,切換的代碼在線程級別復用起來,而不用再單獨設計線程。也就說,以后再創建線程,只需要創建pcb,然后指向同一個進程地址空間,線程的管理就可以復用進程的管理代碼。這就是linux中線程的實現方案。

那就有一個問題,在CPU看來,一個pcb到底是進程還是線程,或者說CPU要不要區分一個pcb是進程還是線程。答案很明顯,CPU不需要區分進程和線程,CPU只需要根據pcb的地址空間來執行代碼即可。也就是現在CPU拿到一個pcb,其執行流是小于等于進程的(當該進程內有多個pcb,其執行流小于進程;當該進程只有一個pcb,其執行流等于該進程)。那現在什么是進程?進程 = 該進程的所有pcb + 地址空間 + 頁表 + 代碼和數據。與以往進程的區別就是,現在進程內部有多個執行流,以前進程內部只有一個執行流。
在這里插入圖片描述

紅色框內的所有東西之和就是進程。
現在我們就可以理解進程是承擔分配系統資源的基本實體,線程是參與資源分配。進程創建要申請系統資源,來創建一個pcb,地址空間,頁表,代碼和數據,線程創建就是創建一個pcb來分配該進程內部的資源(劃分地址空間)。實際上,在linux中并沒有真正意義的線程,只是用進程的數據結構來模擬的線程。這種描述執行流的pcb就是輕量級進程(LWP light wigth process 執行流小于等于進程)。那以后,CPU調度就不再是進程,而是一個一個的輕量級進程(pcb),也就是線程是CPU調度的基本單位。


線程比進程更輕量化的原因

  • 線程創建銷毀更簡單,線程只需創建銷毀一個pcb來參與資源的分配,而進程創建銷毀不僅僅只需要一個pcb
  • 線程在地址空間中運行
  • 線程調度更簡單;在同一進程內,線程之間切換是不需要更改地址空間和頁表,只需要將運行中產生的臨時數據進行切換即可,也就是只需切換少量的上下文數據。但這不是主要原因,在cpu內有一個大的存儲空間cache用來進行數據的緩存(熱數據),cache在緩存中是以進程為單位的,那理論上,線程做切換,就不需要切換cache,著就是線程切換更簡單。因為有局部性原理(如當前訪問的代碼附近的代碼,有可能是下次要訪問的代碼)給預加載機制,提供理論基礎,
    在這里插入圖片描述

線程代碼的簡單示例

經過上面的描述,我們已經對線程有了一定的理解,下面就讓我們在代碼層面上來看看。

#include <iostream>
#include <pthread.h>
#include <unistd.h>
#include <sys/types.h>// 新線程
void *ThreadRountine(void *arg)
{const char *threadname = (const char *)arg;while (true){std::cout << "I am a new thread: " << threadname << ", pid: " << getpid() << std::endl;sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRountine, (void*)"thread 1");// 主線程while (true){std::cout << "I am main thread" << ", pid: " << getpid() <<std::endl;sleep(1);}return 0;
}

上面代碼,我們創建了一個新線程,并讓主線程和新線程都執行死循環。
在這里插入圖片描述

在這里插入圖片描述

不出所料,只有一個進程在執行,主線程和新線程都在執行,并且pid相同(在同一個進程內)。那如何查看線程呢? ps -aL查看。

在這里插入圖片描述
果然有兩個線程,其中主線程的LWP 和 PID是相同的。在操作系統中,是通過LWP來識別不同的輕量級進程的。


#include <iostream>
#include <pthread.h>
#include <unistd.h>
#include <sys/types.h>int gnt = 100;
// 新線程
void *ThreadRountine(void *arg)
{const char *threadname = (const char *)arg;while (true){std::cout << "I am a new thread: " << threadname << ", gnt = " << gnt << ", &gnt" << &gnt << std::endl;gnt--;sleep(1);}
}int main()
{pthread_t tid;pthread_create(&tid, nullptr, ThreadRountine, (void*)"thread 1");// 主線程while (true){std::cout << "I am main thread" << ", gnt = " << gnt << ", &gnt" << &gnt  <<std::endl;sleep(1);}return 0;
}

上述代碼,我們創建了兩個線程,其中新線程式gnt–,兩個線程都打印gnt的值和地址。
在這里插入圖片描述
在這里插入圖片描述
可以發現兩個線程共享全局變量gnt。


總結

以上就是我對于線程概念的理解和知識總結。

在這里插入圖片描述

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/717467.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/717467.shtml
英文地址,請注明出處:http://en.pswp.cn/news/717467.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

VS Code 的粘性滾動預覽 - 類似于 Excel 的凍結首行

VS Code 的粘性滾動預覽 - 類似于 Excel 的凍結首行功能&#xff0c;即滾動 UI 顯示當前源代碼范圍。便于在代碼行數比較多的時候更好的知道自己所在的位置。粘性滾動UI 顯示用戶在滾動期間所處的范圍&#xff0c;將顯示編輯器頂部所在的類/接口/命名空間/函數/方法/構造函數&a…

4、Linux-常用命令(二)

目錄 一、搜索命令 1、命令搜索命令 2、文件搜索命令find。格式&#xff1a;find [搜索范圍] [搜索條件]。 3、字符串搜索命令grep 二、幫助命令 1、man【詳細的幫助】 2、--help【簡要的幫助】 三、壓縮與解壓命令 1、.zip格式 2、.gz格式 3、打包 四、關機和重啟命…

【大廠AI課學習筆記NO.57】(10)分類任務的評價指標

我們實際做的是一個分類任務。 在人工智能深度學習項目中&#xff0c;分類任務是指一種特定的任務類型&#xff0c;即預測結果是離散值的任務。具體來說&#xff0c;分類任務的目標是將輸入數據劃分到不同的類別中。這些類別可以是二分類&#xff08;如垃圾郵件分類&#xff0c…

理解這幾個安全漏洞,你也能做安全測試

01 短信炸彈 1、漏洞描述 短信轟炸攻擊是常見的一種攻擊&#xff0c;攻擊者通過網站頁面中所提供的發送短信驗證碼的功能處&#xff0c;通過對其發送數據包的獲取后&#xff0c;進行重放&#xff0c;如果服務器短信平臺未做校驗的情況時&#xff0c;系統會一直去發送短信&…

函數式響應式編程(FRP):構筑靈活動態的應用程序

FRP&#xff08;Functional Reactive Programming&#xff0c;函數式響應式編程&#xff09;是一個編程范式&#xff0c;它結合了函數式編程和響應式編程的原則&#xff0c;用于處理時間變化的數據和響應性系統。FRP 讓開發者能夠以聲明式地方式來構建響應用戶輸入、網絡請求或…

【vue3 路由使用與講解】vue-router : 簡潔直觀的全面介紹

# 核心內容介紹 路由跳轉有兩種方式&#xff1a; 聲明式導航&#xff1a;<router-link :to"...">編程式導航&#xff1a;router.push(...) 或 router.replace(...) &#xff1b;兩者的規則完全一致。 push(to: RouteLocationRaw): Promise<NavigationFailur…

JVM內部世界(內存劃分,類加載,垃圾回收)

&#x1f495;"Echo"&#x1f495; 作者&#xff1a;Mylvzi 文章主要內容&#xff1a;JVM內部世界(內存劃分,類加載,垃圾回收) 關于JVM的學習主要掌握三方面: JVM內存區的劃分類加載垃圾回收 一.JVM內存區的劃分 當一個Java進程開始執行時,JVM會首先向操作系統申…

實例驅動計算機網絡

文章目錄 計算機網絡的層次結構應用層DNSHTTP協議HTTP請求響應過程 運輸層TCP協議TCP協議面向連接實現TCP的三次握手連接TCP的四次揮手斷開連接 TCP協議可靠性實現TCP的流量控制TCP的擁塞控制TCP的重傳機制 UDP協議 網際層IP協議&#xff08;主機與主機&#xff09;IP地址的分類…

php 讀取文件并以文件方式下載

if (!file_exists($filename)){//判斷能否獲取這個文件header("Content-type: text/html; charset=utf-8");echo "File not found!";exit

【創作回顧】17個月崢嶸創作史

#里程碑專區#、#創作者紀念日# 還記得 2022 年 10 月 05 日&#xff0c;我在CSDN撰寫了第 1 篇博客——《關于測試工程師瓶頸和突圍的一個思考》&#xff0c;也是我在全網發布的第一篇技術文章。 回想當時&#xff0c;這一篇的誕生過程并不輕松&#xff0c;不像是一篇網絡文章…

【計算機網絡】深度學習HTTPS協議

&#x1f493; 博客主頁&#xff1a;從零開始的-CodeNinja之路 ? 收錄文章&#xff1a;【計算機網絡】深度學習HTTPS協議 &#x1f389;歡迎大家點贊&#x1f44d;評論&#x1f4dd;收藏?文章 目錄 一:HTTPS是什么二:HTTPS的工作過程三:對稱加密四:非對稱加密五:中間人攻擊1…

【web | CTF】BUUCTF [HCTF 2018]WarmUp

天命&#xff1a;這題本地php代碼是無法復現的 首先打開網站&#xff0c;啥也沒有&#xff0c;查看源碼 發現文件&#xff0c;打開訪問一下看看&#xff0c;發現是代碼審計 <?phphighlight_file(__FILE__);class emmm{public static function checkFile(&$page){$whit…

【學習總結】什么是DoS和DDoS

[Q&A] 什么是DoS DoS 是 “Denial of Service”&#xff08;拒絕服務&#xff09;的縮寫&#xff0c;它是一種網絡攻擊方式&#xff0c;其目的是使目標計算機或網絡資源無法為合法用戶提供正常的服務。通過向目標系統發送大量請求、消耗其帶寬、處理器或內存等資源&#…

13 雙口 RAM IP 核

雙口 RAM IP 核簡介 雙口 RAM IP 核有兩個端口&#xff0c;它又分為偽雙端口 RAM 和真雙端口 RAM&#xff0c;偽雙端口 RAM 一個端口只能讀&#xff0c;另一個端口只能 寫&#xff0c;真雙端口 RAM 兩個端口都可以進行讀寫操作。同時對存儲器進行讀寫操作時就會用到雙端口 RAM…

unity-1

創建游戲對象&#xff08;游戲物體&#xff09; 可通過unity中的菜單欄中的Gameobject創建&#xff1b;也可在Hierarchy&#xff08;層級&#xff09;中創建&#xff0c; 雙擊即可居中看到。 在Hierarchy空白處右鍵即可看到&#xff0c;能創建游戲對象。 在Scene框中&#x…

BioTech - ADMET的性質預測 概述

歡迎關注我的CSDN&#xff1a;https://spike.blog.csdn.net/ 本文地址&#xff1a;https://blog.csdn.net/caroline_wendy/article/details/136438192 ADMET&#xff0c;即 Absorption、Distribution、Metabolism、Excretion、Toxicity&#xff0c;吸收、分布、代謝、排泄、毒性…

題目 1629: 藍橋杯算法訓練VIP-接水問題

題目描述: 學校里有一個水房&#xff0c;水房里一共裝有m個龍頭可供同學們打開水&#xff0c;每個龍頭每秒鐘的供水量相等&#xff0c;均為1。現在有n名同學準備接水&#xff0c;他們的初始接水順序已經確定。將這些同學按接水順序從1到n編號&#xff0c;i號同學的接水量為wi。…

Linux shell:補充命令的使用

目錄 一.導讀 二.正文 三.結語 一.導讀 上一篇介紹了腳本的簡單概念以及使用&#xff0c;現在補充一些命令。 二.正文 目前處于全局目錄&#xff0c;通過mkdir創建名我為day01的文件。 通過cd命令day01 切換至day01文件當中。 使用vim文本編輯器文件名&#xff08;firstdir&…

設計模式學習筆記——工廠方法模式

設計模式&#xff08;創建型&#xff09;—— 工廠方法模式 傳統的獲取對象方法&#xff0c;是通過 new 關鍵字獲取一個對象&#xff0c;但是如果多個地方都需要該對象&#xff0c;就需要 new 很多次&#xff0c;這時候如果這個類發生了一些改變&#xff0c;如類名變了&#x…

靜態上下文調用了非靜態上下文

問題描述&#xff1a; static修飾的方法不能調用非static修飾方法 問題原因&#xff1a; 在Java中&#xff0c;靜態方法&#xff08;如main方法&#xff09;可以直接訪問靜態成員&#xff08;包括靜態變量和靜態方法&#xff09;&#xff0c;但不能直接訪問非靜態成員&#…