你真的了解靜態變量、常量的存儲位置嗎?

文章目錄

  • 引言
  • C++對內存的劃分如何落實在Linux上
    • 自由存儲區和堆之間的問題
    • 常量區
    • 靜態存儲區
  • 靜態局部變量
    • 靜態局部變量、靜態全局變量、全局變量的異同
  • macOS系統的測試結果
    • 總結


引言

在動態內存的博客中,我提到:
在這里插入圖片描述
在Linux 內存管理的博客中,我提到:
在這里插入圖片描述

在這里插入圖片描述
盡管都有盡可能完全的描述,并且兩者大致意思沒有沖突。而之所以令我一直感到略有不同,越看越迷糊的原因是:第一張圖講的其實是C++在概念上對內存的劃分,第二張圖講的是Linux對虛擬內存進行的劃分。 前者是概念上的,也是C++程序在運行時會切實執行的,而后者就是在Linux系統上對前者概念的具象化!下面進行進一步分析。


C++對內存的劃分如何落實在Linux上

C++其實將內存劃分為兩種:動態存儲區、靜態存儲區

第一張圖對動態存儲區進行了進一步劃分——堆、棧

而網上其他博客可能還會對動態存儲區進行進一步劃分——堆、棧、自由存儲區。并對靜態存儲區進行進一步劃分——常量存儲區、全局/靜態存儲區。

可謂是五花八門,我們不妨先做個歸攏:

自由存儲區和堆之間的問題

這篇博客分析地很詳細C++ 自由存儲區是否等價于堆?,我引用其中一些內容進行分析:

在概念上我們是這樣區分兩者的:

  • malloc 在堆上分配的內存塊,使用 free 釋放內存。
  • new 所申請的內存則是在自由存儲區上,使用 delete 來釋放。

那么物理上,自由存儲區與堆是兩塊不同的內存區域嗎?它們有可能相同嗎?

基本上,所有的 C++編譯器 默認使用堆來實現自由存儲,也即是缺省的全局運算符 newdelete 也許會按照 mallocfree 的方式來被實現,這時藉由 new 運算符 分配的對象,說它在堆上也對,說它在自由存儲區上也正確。 但程序員也可以通過重載操作符,改用其他內存來實現自由存儲,例如全局變量做的對象池,這時自由存儲區就區別于堆了

總結:

  • 自由存儲C++ 中通過 newdelete 動態分配和釋放對象的抽象概念,而 堆(heap)C語言操作系統 的術語,是操作系統維護的一塊動態分配內存

  • new 所申請的內存區域在 C++ 中稱為自由存儲區。藉由堆實現的自由存儲,可以說 new 所申請的內存區域在堆上

  • 堆與自由存儲區的運作方式不同、訪問方式不同,所以應該被當成不一樣的東西來使用。

如何落實在Linux上?

C++中的自然也就對應Linux中的堆段,而C++中的自由存儲區,如果不主動改用其他內存來實現自由存儲,那么理應也在堆段上。

而正如上面所言,堆段由程序員進行申請和釋放:

int main(){int *pi = new int; // pi指向一個動態分配的、未初始化的無名對象,該對象的地址位于堆上// 而pi的地址位于main函數的棧上
}

C++中的自然對應Linux中的棧段,棧段是進程運行之初(從main函數開始)創建的,進程運行時(main函數中)每調用一個函數就會在棧段上申請一段空間作為棧幀,來管理調用函數的相關信息。

void fun(){int j = 2; // 調用fun時,j存在于fun的棧幀上cout << "hello" << endl;
}
int main(){ // 創建棧段int i = 1; // 存在于棧段上fun(); // 創建棧幀
}

常量區

c++ 中,一個 const 不是必需創建內存空間,而在 c 中,一個 const 總是需要一塊內存空間。

常量分為全局常量和局部常量:

全局常量

是否要為 const全局變量 分配內存空間,取決于這個全局常量的用途,如果是充當著一個值替換(將一個變量名替換為一個值),那么就不分配內存空間,不過當對這個全局常量取地址或者使用 extern 時,會分配內存,存儲在只讀數據段,是不能修改的。

因為全局變量在內存中的位置與全局常量一樣,只不過沒有 read only 屬性,因此在這里也就一并提了,全局常量同樣被分配到數據段上,但是可以修改。

PS:未初始化初始化為0 的全局變量(包括全局常量)被分配在 .bss 段上,已初始化 的被分配在 數據段 上。

局部常量

  1. 對于基礎數據類型,也就是 const int a = 10 這種,編譯器會把它放到符號表中,不分配內存,當對其取地址時,會在棧段分配內存。
  2. 對于基礎數據類型,如果用一個變量初始化 局部常量,如果 const int a = b,那么也是會給 a棧段分配內存。
  3. 對于自定數據類型,比如類對象,那么也會在棧段分配內存。

題外話

  • cconst 默認為外部連接c++const 默認為內部連接
  • c 語言兩個文件中都有 const int a 的時候,編譯器會報重定義的錯誤。
  • 而在 c++ 中則不會,因為 c++ 中的 const 默認是內部連接的。如果想讓 c++ 中的 const 具有外部連接,必須顯式聲明為 extern const int a = 10

示例

const int lx = 5;
// 沒有使用的時候僅保存在符號表
// 使用extern或取地址的時候為其在數據段的只讀部分分配內存
// 個人猜測也有可能在代碼段的.rodata。
int o = 6;class A
{const int lz = 1; // 在棧段分配內存
public:void put() {cout << &lz << endl;}
};int main() {A a;int x = 2; // 對照main中的變量來確定其他常量的位置// 因為我們確定 x 在棧段上// 因此如果其他常量的地址與 x 的地址類似// 則說明其他常量也在棧段上const int z = 1; // 取地址時,會在棧段分配內存const int y = x; // 取地址時,會在棧段分配內存
}

在這里插入圖片描述


靜態存儲區

靜態變量分為:全局靜態變量、局部靜態變量

而關于它們的存儲位置,我在 Linux內存管理 一文中已經說的很詳細了,下面的靜態變量包括全局靜態變量和局部靜態變量:
在這里插入圖片描述


靜態局部變量

猜測下面代碼的輸出結果:

void f(int) {static int i = 0;cout << &i << " " << ++i << endl;
}
void f(double) {static int i = 0;cout << &i << " " << ++i << endl;
}
int main() {f(1);f(1.0);f(1);f(1.0);f(1);
}

答案:
在這里插入圖片描述

這里證明了靜態局部變量的特性:只初始化一次,并且只對定義自己的函數可見。 因此在上面的調用中,并不會出現因為兩個靜態局部變量名字相同而賦值出錯的情況。

靜態局部變量、靜態全局變量、全局變量的異同

  1. 全局變量在整個工程文件內都有效,靜態全局變量只在定義它的文件內有效;
  2. 靜態局部變量只在定義它的函數內有效,且程序僅分配一次內存(只初始化一次),函數返回后,該變量不會消失;
  3. 全局變量靜態變量如果沒有手工初始化,則由編譯器初始化為 0
  4. 靜態局部變量靜態全局變量 共享 數據段(或.BSS段)

macOS系統的測試結果

macOS系統下,測試結果如下:

#include <string>
#include <iostream>using namespace std;const int a = 0;int a1;int a2 = 1;static const int a3 = 2;static int a4 = 2;class A{
public:int b;const int b1 = 1;static const int b2 = 2;static int b3;void put(){cout << "類內變量:" << &b << endl;cout << "類內常量:" << &b1 << endl;}
};const int A::b2;
int A::b3;int main(){int c;const int c1 = 1;static int c2;static const int c3 = 5;cout << "全局常量:" << &a << endl;cout << "全局靜態常量:" << &a3 << endl;cout << "類內靜態常量:" << &A::b2 << endl;cout << "局部靜態常量:" << &c3 << endl;cout << "類內靜態變量:" << &A::b3 << endl;cout << "局部靜態變量:" << &c2 << endl;cout << "全局變量,未初始化:" << &a1 << endl;cout << "全局變量,已初始化:" << &a2 << endl;cout << "全局靜態變量:" << &a4 << endl;cout << "局部變量:" << &c << endl;cout << "局部常量:" << &c1 << endl;A a;a.put();return 0;
}

輸出結果:
在這里插入圖片描述

總結

  • 非靜態非全局類內局部變量 都在 上。
  • 全局】 / 【靜態(不論 局部 還是 類內)】的 常量,都在 .bss 段上。
  • 全局】 / 【靜態(不論 局部 還是 類內)】的 變量,都在 數據段中 .bss 段以外的位置 上。

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

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

相關文章

C++ 泛型編程(二):非類型模板參數,模板特化,模板的分離編譯

文章目錄非類型模板參數函數模板的特化類模板的特化全特化偏特化部分參數特化參數修飾特化模板分離編譯解決方法非類型模板參數 模板的參數分為兩種&#xff1a; 類型參數&#xff1a; 則是我們通常使用的方式&#xff0c;就是在模板的參數列表中在 class 后面加上參數的類型…

Java操作——獲取文件擴展名,去掉文件擴展名

昨天收郵件&#xff0c;得知要參加一個產品部的會議&#xff0c;猜想&#xff0c;也許是因為我做的這個產品demo問題。于是昨天忙活到凌晨3點半&#xff0c;結果早上一來才知道又被調戲了。發郵件的MM把郵件誤發給我了。悲催啊有木有&#xff0c;困啊有木有&#xff01;自己還是…

數據結構 | B樹、B+樹、B*樹

文章目錄搜索結構B樹B樹的插入B樹的遍歷B樹的性能B樹B樹的插入B樹的遍歷B*樹B*樹的插入總結搜索結構 如果我們有大量的數據需要永久存儲&#xff0c;就需要存儲到硬盤之中。但是硬盤的訪問速度遠遠小于內存&#xff0c;并且由于數據量過大&#xff0c;無法一次性加載到內存中。…

MySQL 索引 :哈希索引、B+樹索引、全文索引

文章目錄索引引言常見的索引哈希索引自適應哈希索引B樹索引聚集索引非聚集索引使用方法聯合索引最左前綴匹配規則覆蓋索引全文索引使用方法索引 引言 為什么需要索引&#xff1f; 倘若不使用索引&#xff0c;查找數據時&#xff0c;MySQL必須遍歷整個表。而表越大&#xff0c;…

服裝店怎么引流和吸引顧客 服裝店鋪收銀系統來配合

實體店的同城引流和經營是實體經濟的一個重要的一環&#xff0c;今天我們來分享服裝行業的實體店鋪怎么引流和吸引、留住顧客&#xff0c;并實現復購。大家點個收藏&#xff0c;不然劃走就再也找不到了&#xff0c;另外可以點個關注&#xff0c;下次有新的更好的招&#xff0c;…

約瑟夫環(丟手絹問題)

文章目錄問題描述思路代碼實現問題描述 有 1~N 個數字&#xff0c;從 1~m 依次報數&#xff0c;數到 m 的數字要被刪掉&#xff0c;求最后剩下的數字是&#xff1f; 思路 第一次報數第二次報數1n-m12n-m2……m-2n-2m-1n-1m被刪掉了m11m22……n-1n-1-mnn-m 通過上面的表格&…

MySQL 鎖的相關知識 | lock與latch、鎖的類型、簡談MVCC、鎖算法、死鎖、鎖升級

文章目錄lock與latch鎖的類型MVCC一致性非鎖定讀&#xff08;快照讀&#xff09;一致性鎖定讀&#xff08;當前讀&#xff09;鎖算法死鎖鎖升級lock與latch 在了解數據庫鎖之前&#xff0c;首先就要區分開 lock 和 latch。在數據庫中&#xff0c;lock 和 latch 雖然都是鎖&…

Hibernate使用原生SQL適應復雜數據查詢

HQL盡管容易使用&#xff0c;但是在一些復雜的數據操作上功能有限。特別是在實現復雜的報表統計與計算&#xff0c;以及多表連接查詢上往往無能為力&#xff0c;這時可以使用SQL&#xff08;Native SQL&#xff09;實現HQL無法完成的任務。 1、使用SQL查詢 使用SQL查詢可以通過…

MySQL 存儲引擎 | MyISAM 與 InnoDB

文章目錄概念innodb引擎的4大特性索引結構InnoDBMyISAM區別表級鎖和行級鎖概念 MyISAM 是 MySQL 的默認數據庫引擎&#xff08;5.5版之前&#xff09;&#xff0c;但因為不支持事務處理而被 InnoDB 替代。 然而事物都是有兩面性的&#xff0c;InnoDB 支持事務處理也會帶來一些…

MySQL 事務 | ACID、四種隔離級別、并發帶來的隔離問題、事務的使用與實現

文章目錄事務ACID并發帶來的隔離問題幻讀&#xff08;虛讀&#xff09;不可重復讀臟讀丟失更新隔離級別Read Uncommitted (讀未提交)Read Committed (讀已提交)Repeatable Read (可重復讀)Serializable (可串行化)事務的使用事務的實現Redoundo事務 事務指邏輯上的一組操作。 …

MySQL 備份與主從復制

文章目錄備份主從復制主從復制的作用備份 根據備份方法的不同&#xff0c;備份可劃分為以下幾種類型&#xff1a; 熱備(Hot Backup) &#xff1a; 熱備指的是在數據庫運行的時候直接備份&#xff0c;并且對正在運行的數據庫毫無影響&#xff0c;這種方法在 MySQL 官方手冊中又…

C++ 流的操作 | 初識IO類、文件流、string流的使用

文章目錄前言IO頭文件iostreamfstreamsstream流的使用不能拷貝或對 IO對象 賦值條件狀態與 iostate 類型輸出緩沖區文件流fstream類型文件模式文件光標函數tellg() / tellp()seekg() / seekp()向文件存儲內容/讀取文件內容string流istringstreamostringstream前言 我們在使用 …

Hibernate 更新部分更改的字段 hibernate update

Hibernate 中如果直接使用 Session.update(Object o);或則是Session.updateOrUpdate(Object o); 會把這個表中的所有字段更新一遍。 如&#xff1a; ExperClass4k e new ExperClass4k(); e.setTime(time); e.setQ_num(q_num); e.setK(k); if (str "finch_fix")…

C++ 類的行為 | 行為像值的類、行為像指針的類、swap函數處理自賦值

文章目錄概念行為像值的類行為像指針的類概念引用計數動態內存實現計數器類的swap概念swap實現自賦值概念 行為像值的類和行為像指針的類這兩種說法其實蠻拗口的&#xff0c;這也算是 《CPrimer》 翻譯的缺點之一吧。。。 其實兩者的意思分別是&#xff1a; 行為像值的類&am…

C++ 右值引用 | 左值、右值、move、移動語義、引用限定符

文章目錄C11為什么引入右值&#xff1f;區分左值引用、右值引用move移動語義移動構造函數移動賦值運算符合成的移動操作小結引用限定符規定this是左值or右值引用限定符與重載C11為什么引入右值&#xff1f; C11引入了一個擴展內存的方法——移動而非拷貝&#xff0c;移動較之拷…

且談關于最近軟件測試的面試

前段時間有新的產品需要招人&#xff0c;安排和參加了好幾次面試&#xff0c;下面就談談具體的面試問題&#xff0c;在面試他人的同時也面試自己。 面試問題是參與面試同事各自設計的&#xff0c;我也不清楚其他同事的題目&#xff0c;就談談自己設計的其中2道題。 過去面試總是…

C++ 多態 | 虛函數、抽象類、虛函數表

文章目錄多態虛函數重寫重定義&#xff08;參數不同&#xff09;協變&#xff08;返回值不同&#xff09;析構函數重寫&#xff08;函數名不同&#xff09;final和override重載、重寫、重定義抽象類多態的原理虛函數常見問題解析虛函數表多態 一種事物&#xff0c;多種形態。換…

C++ 運算符重載(一) | 輸入/輸出,相等/不等,復合賦值,下標,自增/自減,成員訪問運算符

文章目錄輸出運算符<<輸入運算符>>相等/不等運算符復合賦值運算符下標運算符自增/自減運算符成員訪問運算符輸出運算符<< 通常情況下&#xff0c;輸出運算符的第一個形參是一個 非常量ostream對象的引用 。之所以 ostream 是非常量是因為向流寫入內容會改變…

C++ 重載函數調用運算符 | 再探lambda,函數對象,可調用對象

文章目錄重載函數調用運算符lambdalambda等價于函數對象lambda等價于類標準庫函數對象可調用對象與function可調用對象function函數重載與function重載函數調用運算符 函數調用運算符必須是成員函數。 一個類可以定義多個不同版本的調用運算符&#xff0c;互相之間應該在參數數…

C++ 運算符重載(二) | 類型轉換運算符,二義性問題

文章目錄類型轉換運算符概念避免過度使用類型轉換函數解決上述問題的方法轉換為 bool顯式的類型轉換運算符類型轉換二義性重載函數與類型轉換結合導致的二義性重載運算符與類型轉換結合導致的二義性類型轉換運算符 概念 類型轉換運算符&#xff08;conversion operator&#…