自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

關(guān)鍵字Static的使用詳解

開(kāi)發(fā) 前端
要想搞清楚關(guān)鍵字static的使用方法,必須首先搞清楚,可執(zhí)行程序段的分類(lèi)以及各段在內(nèi)存區(qū)的邏輯地址的映射。

[[379789]]

 粉絲提問(wèn)

粉絲問(wèn)題,總結(jié)一下:關(guān)鍵字static的使用方法。


問(wèn)題

要想搞清楚關(guān)鍵字static的使用方法,必須首先搞清楚,可執(zhí)行程序段的分類(lèi)以及各段在內(nèi)存區(qū)的邏輯地址的映射。

本文配套視頻,請(qǐng)見(jiàn)次條文章《【視頻講解】C語(yǔ)言static關(guān)鍵詞》

一、可執(zhí)行程序內(nèi)存分配

1. 可執(zhí)行程序程序分段

一個(gè)程序的3個(gè)基本段:text段,data段,bss段


BSS BSS(Block Started by Symbol)通常是指用來(lái)存放程序中未初始化的全局變量和靜態(tài)變量的一塊內(nèi)存區(qū)域。

特點(diǎn)是:可讀寫(xiě)的,在程序執(zhí)行之前BSS段會(huì)自動(dòng)清0。

所以,未初始的全局變量在程序執(zhí)行之前已經(jīng)成0了。

注意和數(shù)據(jù)段的區(qū)別,BSS存放的是未初始化的全局變量和靜態(tài)變量,數(shù)據(jù)段存放的是初始化后的全局變量和靜態(tài)變量。

UNIX下可使用size命令查看可執(zhí)行文件的段大小信息。如size a.out。

數(shù)據(jù)段.data 存放在編譯階段(而非運(yùn)行時(shí))就能確定的數(shù)據(jù),可讀可寫(xiě)。

也就是通常所說(shuō)的靜態(tài)存儲(chǔ)區(qū),賦了初值的全局變量和賦初值的靜態(tài)變量存放在這個(gè)區(qū)域,常量也存在這個(gè)區(qū)域。數(shù)據(jù)段,代碼段在程序運(yùn)行之前就已經(jīng)確定了的。

代碼段.text 代碼段通常是指用來(lái)存放程序執(zhí)行代碼的一塊內(nèi)存區(qū)域。

這部分區(qū)域的大小在程序運(yùn)行前就已經(jīng)確定,并且內(nèi)存區(qū)域通常屬于只讀, 某些架構(gòu)也允許代碼段為可寫(xiě),即允許自修改程序。

在代碼段中,也有可能包含一些只讀的常數(shù)變量,例如字符串常量等。

text段在編譯時(shí)確定,內(nèi)存中被映射為只讀,但date段與bss段是可寫(xiě)的。

2. c語(yǔ)言五大內(nèi)存分區(qū)

1.棧區(qū)(堆棧區(qū)stack)

堆棧是由編譯器自動(dòng)分配釋放,存放函數(shù)的參數(shù)和局部變量的值(auto類(lèi)型),操作方式類(lèi)似于數(shù)據(jù)結(jié)構(gòu)中的棧。棧的申請(qǐng)是由系統(tǒng)自動(dòng)分配,如在函數(shù)內(nèi)部申請(qǐng)一個(gè)局部變量int h,同時(shí)判斷所申請(qǐng)空間是否小于棧的剩余空間,如果小于則為其開(kāi)辟空間,為程序提供內(nèi)存,否則將報(bào)異常提示棧溢出。

2.堆(heap)

堆一般由程序員分配釋放,若程序員不釋放,程序結(jié)束可能由OS回收。

它與數(shù)據(jù)結(jié)構(gòu)中的堆是兩回事,分配方式類(lèi)似于鏈表,申請(qǐng)則是程序員自己操作使用malloc或new。

申請(qǐng)過(guò)程比較復(fù)雜,當(dāng)系統(tǒng)收到程序的申請(qǐng)時(shí),會(huì)遍歷記錄空閑內(nèi)存地址的鏈表,以求尋找第一個(gè)空間大于所申請(qǐng)空間的堆節(jié)點(diǎn),然后將該節(jié)點(diǎn)從空閑節(jié)點(diǎn)鏈表中刪除,并將該節(jié)點(diǎn)的空間分配給程序,有些情況下,新申請(qǐng)的內(nèi)存塊的首地址記錄本次分配的內(nèi)存塊的大小,這樣在free()時(shí)能正確的釋放內(nèi)存空間。

3.全局靜態(tài)存儲(chǔ)區(qū)

全局變量與靜態(tài)變量的存儲(chǔ)是放在一塊的,初始化的全局變量與靜態(tài)變量存放在一塊區(qū)域,未初始化的全局變量與未初始化的靜態(tài)變量存放在相鄰的另一塊區(qū)域。

4.文字常量區(qū)

常量字符串就是放在該部分,只讀存儲(chǔ)區(qū),程序結(jié)束后由系統(tǒng)釋放

5.程序代碼區(qū)

存放程序的二進(jìn)制代碼區(qū)。


兩者之間區(qū)別是:代碼段,數(shù)據(jù)段,堆棧段是cpu級(jí)別的概念,五大分區(qū)屬于語(yǔ)言級(jí)別的概念,兩者是不同的概念。

3. 可執(zhí)行程序內(nèi)存空間與邏輯地址空間的映射與劃分

 

左邊是UNIX系統(tǒng)的執(zhí)行文件,右邊是進(jìn)程對(duì)應(yīng)的邏輯地址空間的劃分情況

4. 舉例


二、static 變量

static變量主要區(qū)分靜態(tài)全局變量和全局變量、局部變量和靜態(tài)局部變量之間的區(qū)別。

1. 靜態(tài)全局變量、全局變量

靜態(tài)全局變量、全局變量的區(qū)別主要通過(guò)生存周期和作用域來(lái)區(qū)別。

  • a.靜態(tài)全局變量和全局變量均存放在數(shù)據(jù)段.data中;
  • b. 靜態(tài)局部變量在函數(shù)內(nèi)定義,生存期為整個(gè)源程序,但作用域與自動(dòng)變量相同,只能在定義該變量的函數(shù)內(nèi)使用。退出該函數(shù)后, 盡管該變量還繼續(xù)存在,但不能使用它。
  • c. 對(duì)基本類(lèi)型的靜態(tài)局部變量若在說(shuō)明時(shí)未賦以初值,則系統(tǒng)自動(dòng)賦予0值。而對(duì)自動(dòng)變量不賦初值,則其值是不定的。
  • d.全局變量本身就是靜態(tài)存儲(chǔ)方式, 靜態(tài)全局變量當(dāng)然也是靜態(tài)存儲(chǔ)方式。但是他們的作用域,非靜態(tài)全局 變量的作用域是整個(gè)源程序(多個(gè)源文件可以共同使用);而靜態(tài)全局變量則限制了其作用域, 即只在定義該變量的源文件內(nèi)有效, 在同一源程序的其它源文件中不能使用它。

全局變量實(shí)例

以下是b.c 和 a.c源代碼

 

全局變量

編譯

  1. gcc a.c b.c 

執(zhí)行結(jié)果:

 

執(zhí)行結(jié)果

由編譯結(jié)果可知,文件a.c可以訪(fǎng)問(wèn)到b.c文件中的靜態(tài)全局變量b。

靜態(tài)全局變量實(shí)例


編譯結(jié)果

 

編譯結(jié)果

由編譯結(jié)果可知,文件a.c無(wú)法訪(fǎng)問(wèn)到b.c文件中的靜態(tài)全局變量b,所以編譯報(bào)錯(cuò)。

2. 靜態(tài)局部變量、局部變量

靜態(tài)局部變量、局部變量的區(qū)別主要通過(guò)生存周期和作用域來(lái)區(qū)別。

靜態(tài)局部變量存放在數(shù)據(jù)段.data中,局部變量在棧中;靜態(tài)局部變量和局部變量都只能在函數(shù)體內(nèi)部才可以訪(fǎng)問(wèn)。

函數(shù)每次訪(fǎng)問(wèn)的靜態(tài)局部變量,該變量的值為最后一次訪(fǎng)問(wèn)修改后的值。

舉例:

  1.  1 #include <stdio.h> 
  2.  2  
  3.  3  
  4.  4 void func() 
  5.  5 { 
  6.  6     int aa = 11; 
  7.  7  
  8.  8     printf("aa= %d \n",aa++); 
  9.  9  
  10. 10 } 
  11. 11  
  12. 12 int main(int argc, char **argv) 
  13. 13 { 
  14. 14  
  15. 15     func();                                                                                     
  16. 16     func(); 
  17. 17  
  18. 18     return 0; 
  19. 19 } 

 

對(duì)于普通的局部變量,每次調(diào)用的時(shí)候,都會(huì)在棧里初始化1次,

  1. 1 #include <stdio.h> 
  2.  2  
  3.  3  
  4.  4 void func() 
  5.  5 { 
  6.  6     static int aa = 11; 
  7.  7                                                                                                 
  8.  8     printf("aa= %d \n",aa++); 
  9.  9  
  10. 10 } 
  11. 11  
  12. 12 int main(int argc, char **argv) 
  13. 13 { 
  14. 14      
  15. 15     func(); 
  16. 16     func(); 
  17. 17  
  18. 18     return 0; 
  19. 19 } 

 

函數(shù)中靜態(tài)變量aa 只初始化一次,每次訪(fǎng)問(wèn)的值應(yīng)該是上一次調(diào)用到該函數(shù)時(shí)最后處理的結(jié)果,

三、static 函數(shù)

1. 概念:

在函數(shù)的返回類(lèi)型前加上關(guān)鍵字static,函數(shù)就被定義成為靜態(tài)函數(shù)。

函數(shù)的定義和聲明默認(rèn)情況下是extern的,但靜態(tài)函數(shù)只是在聲明他的文件當(dāng)中可見(jiàn),不能被其他文件所用。

static函數(shù)(也叫內(nèi)部函數(shù))只能被本文件中的函數(shù)調(diào)用,而不能被同一程序其它文件中的函數(shù)調(diào)用。

區(qū)別于一般的非靜態(tài)函數(shù)(外部函數(shù)) static在c里面可以用來(lái)修飾變量,也可以用來(lái)修飾函數(shù)。

先看用來(lái)修飾變量的時(shí)候。變量在c里面可分為存在全局?jǐn)?shù)據(jù)區(qū)、棧和堆里。

其實(shí)我們平時(shí)所說(shuō)的堆棧是棧而不包含堆,不要弄混。

2. 定義靜態(tài)函數(shù)的好處:

  • <1>其他文件中可以定義相同名字的函數(shù),不會(huì)發(fā)生沖突,不用擔(dān)心自己定義的函數(shù),是否會(huì)與其它文件中的函數(shù)同名,因?yàn)橥矝](méi)有關(guān)系。
  • <2> 靜態(tài)函數(shù)不能被其他文件所用。存儲(chǔ)說(shuō)明符auto,register,extern,static,對(duì)應(yīng)兩種存儲(chǔ)期:自動(dòng)存儲(chǔ)期和靜態(tài)存儲(chǔ)期。
  • <3> 統(tǒng)計(jì)次數(shù)功能 聲明函數(shù)的一個(gè)局部變量,并設(shè)為static類(lèi)型,作為一個(gè)計(jì)數(shù)器,這樣函數(shù)每次被調(diào)用的時(shí)候就可以進(jìn)行計(jì)數(shù)。這是統(tǒng)計(jì)函數(shù)被調(diào)用次數(shù)的最好的辦法,因?yàn)檫@個(gè)變量是和函數(shù)息息相關(guān)的,而函數(shù)可能在多個(gè)不同的地方被調(diào)用,所以從調(diào)用者的角度來(lái)統(tǒng)計(jì)比較困難。
  • <4> 靜態(tài)函數(shù)會(huì)被自動(dòng)分配在一個(gè)一直使用的存儲(chǔ)區(qū),直到退出應(yīng)用程序?qū)嵗?,避免了調(diào)用函數(shù)時(shí)壓棧出棧,速度快很多。

舉例

a.c

  1.  1 #include <stdio.h> 
  2.  2  
  3.  3 void func(); 
  4.  4  
  5.  5 int main(int argc, char **argv) 
  6.  6 { 
  7.  7      
  8.  8     func();                                                                                                         
  9.  9  
  10. 10     return 0; 
  11. 11 } 

b.c

  1. 1 #include <stdio.h> 
  2. 2  
  3. int b = 10; 
  4. 4  
  5. 5  
  6. static void func()                                                                                                  
  7. 7 { 
  8. 8     printf("in func b =%d\n",b); 
  9. 9 } 

 

編譯

由編譯結(jié)果可知,a文件訪(fǎng)問(wèn)不到b文件中的靜態(tài)函數(shù)func。

 

責(zé)任編輯:姜華 來(lái)源: 一口Linux
相關(guān)推薦

2023-03-09 07:38:58

static關(guān)鍵字狀態(tài)

2022-02-17 08:31:38

C語(yǔ)言staic關(guān)鍵字

2009-06-25 10:33:53

StaticJava類(lèi)

2020-12-09 05:19:35

static關(guān)鍵字Java

2011-07-14 23:14:42

C++static

2022-11-12 18:32:50

Golangomitemptyjson

2013-01-30 10:12:14

Pythonyield

2009-09-02 09:24:03

C# this關(guān)鍵字

2010-02-02 14:27:54

C++ static關(guān)

2009-09-28 11:34:49

Javascript

2023-09-22 22:27:54

autoC++11

2011-06-14 13:26:27

volatile

2024-04-08 11:35:34

C++static關(guān)鍵字

2015-11-10 16:10:22

C語(yǔ)言StaticConst

2012-06-02 00:53:39

Javafinally

2009-09-07 18:53:46

static關(guān)鍵字

2011-04-21 16:57:56

staticextern

2021-07-27 07:31:16

單例模式關(guān)鍵字

2010-07-23 14:32:43

SQL Server

2010-02-05 15:51:06

C++ explici
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)