【Linux系統化學習】進程地址空間 | 虛擬地址和物理地址的關系

=========================================================================

個人主頁點擊直達:小白不是程序媛

Linux專欄:Linux系統化學習

代碼倉庫:Gitee

=========================================================================

目錄

虛擬地址和物理地址

頁表

進程地址空間

進程地址空間存在的意義


虛擬地址和物理地址

我們在學習C/C++的時候肯定都見過下面這張有關于內存分布的圖片:

在來段代碼理解感受下:

  1 #include<stdio.h>2 #include<stdlib.h>3 //未初始化常量4 int un_gval;5 //初始化常量6 int init_gval=100;7 int main()8 {9     //代碼區地址10     printf("code addr: %p\n",main);11     //字符常量12    const char *str="hellolinux!";13 14     //常量區地址15     printf("read only char addr : %p\n",str);16     //已初始化全局數據區17     printf("init global value addr: %p\n",&init_gval);18     //未初始化全局數據區19     printf("uninit global value addr: %p\n",&un_gval);20 21     char *heap1=(char*)malloc(100);22     char *heap2=(char*)malloc(100);23     char *heap3=(char*)malloc(100);24     char *heap4=(char*)malloc(100);25     static int a=0;26     printf("heap1 addr:%p\n",heap1);                                                                                                                                                                    27     printf("heap2 addr:%p\n",heap2);28     printf("heap3 addr:%p\n",heap3);29     printf("heap4 addr:%p\n",heap4);30 31     printf("stack addr:%p\n",&str);32     printf("stack addr:%p\n",&heap1);33     printf("stack addr:%p\n",&heap2);34     printf("stack addr:%p\n",&heap3);35     printf("stack addr:%p\n",&heap4);36     printf("a addr:%p\n",&a);37 38     return 0;39 }

通過上面這段代碼,我們好像不僅驗證了上面的空間分布圖片,而且還發現了棧區和堆區相向而生的內存開辟特點。

上兩篇文章我們介紹了命令行參數和環境變量,其實這兩個就儲存在棧區之上的空間,來段代碼驗證下:

    1 #include<stdio.h>
W>  2 int main(int argc , char *argv[], char *env[])3 {4     int i=0;5     printf("i addr:%p\n",&i);                              6     for(;argv[i];i++)7     {8         printf("argv[%d]:%p\n",i,argv[i]);9     }10     for(i=0;env[i];i++)11     {12         printf("env[%d]:%p\n",i,env[i]);13     }14     return 0;15 }~

?

驗證完這些,話說回來其實我們之前學的對內存的概念就上面所介紹的內容其實都不是真正意義上的內存是虛擬內存,不是我們真正意義上的內存物理地址。

  1 #include<stdio.h>2 #include<unistd.h>3 #include<stdlib.h>                                                                                                                                                                     4 int g_val = 100;5 6 int main()7 {8     pid_t id = fork();9     if(id == 0)10     {11         //child12         int cnt = 5;13         while(1)14         {15             printf("child, Pid: %d, Ppid: %d, g_val: %d, &g_val=%p\n", getpid(), getppid(), g_val, &g_val);16             sleep(1);17             if(cnt == 0)18             {19                 g_val=200;20                printf("child change g_val: 100->200\n");21             }22             cnt--;23         }24     }25     else26     {27         //father28         while(1)29         {30             printf("father, Pid: %d, Ppid: %d, g_val: %d, &g_val=%p\n", getpid(), getppid(), g_val, &g_val);31             sleep(1);32         }33     }34 35     sleep(100);36     return 0;37 }

我們發現,父子進程,輸出地址是一致的,但是變量內容不一樣!能得出如下結論:

  • 變量內容不一樣,所以父子進程輸出的變量絕對不是同一個變量?
  • 但地址值是一樣的,說明,該地址絕對不是物理地址!
  • 在Linux地址下,這種地址叫做 虛擬地址
  • 我們在用C/C++語言所看到的地址,全部都是虛擬地址!物理地址,用戶一概看不到,由OS統一管理OS必須負責將 虛擬地址 轉化成 物理地址?

我們再將同一個可執行程序同時運行也會發現兩個進程的獲取到的地址竟然也是一樣的。

?話又說回來,我們的可執行程序運行時肯定會加載到內存中,因此虛擬地址和物理地址一定有關聯,這個關聯就是頁表


頁表

頁表就是將虛擬地址和物理地址聯系起來的一種模型,其中還包括變量是否可以被修改,進程的狀態等諸多信息。

上面的圖就足矣說名問題,同一個變量,地址相同,其實是虛擬地址相同,內容不同其實是被映射到了不同的物理地址!

每個進程的頁表的物理地址存在與CPU中CR3的寄存器中


進程地址空間

進程地址空間其實我們可以使用內存大小的一個范圍,以我們32位總線的機器為例:它的范圍為0000 0000 -> ffff ffff ,也就是0到2^32次方(我們所謂的4GB)。模擬其物理空間大小進行區域劃分后形成棧區、堆區等等的虛擬地址,操作系統通過結構體將每個區域的起始和結束統計記錄起來,進程的PCB中含有指向這個結構體的指針。

因此,每當新的進程創建時會形成對應的PCB,PCB和PCB中的虛擬地址結構體指針和頁表關聯起來,對真正上的物理地址進行使用。?


進程地址空間存在的意義

  • 讓進程以統一的視角看待內存,所以任意一個進程,可以通過地址空間和頁表可以將亂序的內存數據,變成有序,分門別類的規劃好
  • 存在虛擬地址空間,可以有效的進行進程訪問內存的安全檢查
  • 將進程管理和內存管理進行解耦,通過頁表讓進程映射到不同的物理內存處,從而實現進程的獨立性。

今天對Linux下進程地址空間的介紹分享到這就結束了,希望大家讀完后有很大的收獲,也可以在評論區點評文章中的內容和分享自己的看法。您三連的支持就是我前進的動力,感謝大家的支持!!!

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

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

相關文章

SQL Server 數據庫,多表查詢

4.2使用T-SQL實現多表查詢 前面講述過的所有查詢都是基于單個數據庫表的查詢&#xff0c;如果一個查詢需要對多個表進行操作&#xff0c; 就稱為聯接查詢&#xff0c;聯接查詢的結果集或結果稱為表之間的聯接。 聯接查詢實際上是通過各個表之間共同列的關聯性來查詢數據的&…

微信小程序 bindtap 事件多參數傳遞

在微信小程序中&#xff0c;我們無法直接通過 bindtap"handleClick(1,2,3)" 的方式傳遞參數&#xff0c;而是需要通過自定義屬性 data- 的方式進行傳遞&#xff0c;并在事件回調函數中通過 event.currentTarget.dataset 來獲取這些參數。然而&#xff0c;這種傳參方式…

跨框架解決方案-Mitosis【Context】

Context Mitosis的context必須是&#xff1a; 在自己的文件中創建文件名必須以context.lite.ts結尾默認導出必須是一個返回context對象的函數 // simple.context.lite.ts import { createContext } from builder.io/mitosis;export default createContext({foo: bar,get foo…

速查!軟考出成績了

2023年11月軟考成績出來啦&#xff01;大家趕緊查一下&#xff0c;各科都45分就是通過&#xff01; 01 如何查成績 1、打開“中國計算機技術職業資格網”&#xff0c;網址&#xff1a;https://www.ruankao.org.cn/ 2、點擊↘的“成績查詢”按鈕。 3、輸入“手機號/證件號密碼驗…

光伏儲能IC

光伏儲能IC 電子元器件百科 文章目錄 光伏儲能IC前言一、光伏儲能IC是什么二、光伏儲能IC的類別三、光伏儲能IC的應用實例四、光伏儲能IC的作用原理總結前言 光伏儲能IC可以提高光伏發電和儲能系統的效率和穩定性,同時提供智能化的管理和控制功能。它在光伏發電和儲能領域具有…

船舶機電設備智能故障診斷系統

1 船舶電氣設備故障診斷專家系統 體系結構 整個系統的體系結構采用瀏覽器&#xff0f; 服務器 &#xff08;B&#xff0f; S&#xff09; 三層體系結構 &#xff08; 如圖 1 所示 &#xff09;。B&#xff0f;S 模式是一種 以 Web 技術為基礎的系統平臺模式 。 把傳統…

【GEE】時間序列多源遙感數據隨機森林回歸預測|反演|驗證|散點圖|完整代碼

實驗介紹 分類和回歸之間的主要區別在于&#xff0c;在分類中&#xff0c;我們的預測目標是離散的類別&#xff0c;而在回歸中&#xff0c;預測目標是連續的預測值。 本實驗的研究區域位于佛蒙特州的埃塞克斯郡&#xff0c;使用訓練數據來模擬土壤氧化還原深度&#xff0c;然…

Codeforces Round 851 (Div. 2 D:枚舉+組合 Edp)

A - One and Two 相當于找第一個位置前后2的個數相同 #include<bits/stdc.h> using namespace std; const int N 1e610,mod998244353; #define int long long typedef long long LL; typedef pair<int, int> PII; const long long inf1e17; int n,m,k; int a[N]…

有哪些值得分享的銷售拓客技巧?

拓客對于銷售的重要性 拓客&#xff08;Toker&#xff09;是一個商業上的名詞&#xff0c;核心就是提高售前服務、市場推廣的水平&#xff0c;從而挖掘出潛在客戶的隱形需求&#xff08;或稱軟需求&#xff09;。 拓客的核心&#xff0c;其實就是提高售前服務、市場推廣的水平…

如何部署自己的服務渲染頁面為Pdf文檔

前言 相信大家都覺得官方發布的文檔生成模塊https://docs.mendix.com/appstore/modules/document-generation/很有用&#xff0c;它能把Mendix頁面像素級導出到Pdf文件中&#xff0c;這對于歸檔等業務非常有價值。但部署依賴公有云提供的渲染服務&#xff0c;而中國本土用戶對…

折半查找(數據結構實訓)

題目&#xff1a; 標準輸入輸出 題目描述&#xff1a; 實現折半查找。要求查找給定的值在數據表中相應的存儲位置。本題目假定輸入元素均按非降序輸入。 輸入&#xff1a; 輸入包含若干個測試用例&#xff0c;第一行為測試用例個數k。每個測試用例占3行&#xff0c;其中第一行為…

初識人工智能,一文讀懂過擬合欠擬合和模型壓縮的知識文集(3)

&#x1f3c6;作者簡介&#xff0c;普修羅雙戰士&#xff0c;一直追求不斷學習和成長&#xff0c;在技術的道路上持續探索和實踐。 &#x1f3c6;多年互聯網行業從業經驗&#xff0c;歷任核心研發工程師&#xff0c;項目技術負責人。 &#x1f389;歡迎 &#x1f44d;點贊?評論…

SQL存儲過程和視圖

1 存儲過程 存儲過程是事先編寫好、存儲在數據庫中的一組SQL命令集合。用來完成對數據庫的指定操作。 1.1 優缺點 優點&#xff1a; 1&#xff09;提高系統性能。創建時進行編譯&#xff0c;隨后存放在數據庫服務器的過程高速緩存中&#xff0c;之后不需要再次執行分析和編…

uniapp app將base64保存到相冊,uniapp app將文件流保存到相冊

如果是文件流可以先轉base64詳情見>uniapp 顯示文件流圖片-CSDN博客 onDown(){let base64 this.qrcodeUrl ; // base64地址const bitmap new plus.nativeObj.Bitmap("test");bitmap.loadBase64Data(base64, function() {const url "_doc/" new Dat…

Backend - Dbeaver

目錄 一、說明 二、下載并安裝 &#xff08;一&#xff09;官網下載 &#xff08;二&#xff09;安裝 三、使用 &#xff08;一&#xff09;操作步驟 &#xff08;二&#xff09;相關問題&#xff1a;無法加載驅動類oracle.jdbc.oracledriver 1. 新建驅動 2. 再重新連接數據庫 …

PyTorch2.0環境搭建

一、安裝python并配置環境變量 1、打開python官網&#xff0c;下載并安裝 Welcome to Python.org 下載 尋找版本&#xff1a;推薦使用3.9版本&#xff0c;或其他表中顯示為安全&#xff08;security&#xff09;的版本 安裝&#xff1a;&#xff08;略&#xff09; 2、配置環…

數據增強改進,實現檢測目標copypaste,增加目標數據量,提升精度

???YOLOv8實戰寶典--星級指南:從入門到精通,您不可錯過的技巧 ??-- 聚焦于YOLO的 最新版本, 對頸部網絡改進、添加局部注意力、增加檢測頭部,實測漲點 ?? 深入淺出YOLOv8:我的專業筆記與技術總結 ??-- YOLOv8輕松上手, 適用技術小白,文章代碼齊全,僅需 …

python圣誕樹代碼編程

以下是一個簡單的Python圣誕樹代碼&#xff1a; def draw_tree(height): for i in range(height): print( * (height - i - 1) * * (2 * i 1)) print( * (height - 1) |)draw_tree(10) 這個函數會繪制一個等腰三角形&#xff0c;其中每一行的星號數量從1開…

Java基礎知識

JVM&#xff0c;JRE&#xff0c;JDK JVM 運行Java字節碼的機器 JRE Java運行時環境&#xff0c;包括JVM&#xff0c;Java類庫&#xff0c;運行時類庫&#xff0c;國際化支持&#xff0c;安全管理器&#xff0c;啟動器等 比JVM多的內容 Java類庫&#xff1a;提供大量已經實…

【TiDB理論知識09】TiFlash

一 TiFlash架構 二 TiFlash 核心特性 TiFlash 主要有 異步復制、一致性、智能選擇、計算加速 等幾個核心特性。 1 異步復制 TiFlash 中的副本以特殊角色 (Raft Learner) 進行異步的數據復制&#xff0c;這表示當 TiFlash 節點宕機或者網絡高延遲等狀況發生時&#xff0c;Ti…