程序在運(yùn)行的時(shí)候,其內(nèi)存的來(lái)源主要通過(guò)三種方法: 棧 堆 數(shù)據(jù)段,總體上來(lái)講棧是一般用來(lái)存放小內(nèi)存的局部變量,堆內(nèi)存和數(shù)據(jù)段的屬性很像,在使用的的時(shí)候,如果這個(gè)變量是伴隨程序一直存在則使用全局變量,也就是放在數(shù)據(jù)段,如果一個(gè)變量使用完了就沒(méi)用了,那么就適合用堆內(nèi)存(先申請(qǐng),然后釋放即可),
目前創(chuàng)新互聯(lián)已為近千家的企業(yè)提供了網(wǎng)站建設(shè)、域名、網(wǎng)絡(luò)空間、網(wǎng)站托管、服務(wù)器托管、企業(yè)網(wǎng)站設(shè)計(jì)、興寧網(wǎng)站維護(hù)等服務(wù),公司將堅(jiān)持客戶導(dǎo)向、應(yīng)用為本的策略,正道將秉承"和諧、參與、激情"的文化,與客戶和合作伙伴齊心協(xié)力一起成長(zhǎng),共同發(fā)展。一:棧(stack):
1:棧在使用的時(shí)候是編譯器自動(dòng)分配內(nèi)存空間的,不需要程序員的干涉,其次棧的大小是有限的,所以當(dāng)我們定義的變量需要大片的內(nèi)存的時(shí)候就不適合使用棧,
2:棧存放的是普通變量,棧的在使用的時(shí)候是反復(fù)使用的,所以棧內(nèi)存是臟的,在定義普通變量的時(shí)候,如果不對(duì)變量進(jìn)行初始化,那么變量的值就是隨機(jī)的。
3:棧是先進(jìn)后出的,其內(nèi)存空間是向下增長(zhǎng)的。
二:堆(heap):
1:堆得使用時(shí)是由程序員來(lái)操作的,程序員通過(guò)malloc來(lái)向內(nèi)存申請(qǐng)堆內(nèi)存,使用完以后通過(guò)free來(lái)釋放這部分內(nèi)存,如果這部分內(nèi)存在使用完以后沒(méi)有進(jìn)行內(nèi)存的釋放,那么這部分內(nèi)存管理器就會(huì)認(rèn)為這部分內(nèi)存一直被占用的,體現(xiàn)出來(lái)的就是程序吃內(nèi)存,也就是所謂的內(nèi)存泄漏,如果沒(méi)有釋放內(nèi)存,則被占用的內(nèi)存只有當(dāng)程序結(jié)束以后才會(huì)被釋放。一般需要大片的內(nèi)存時(shí)才使用堆,堆是先進(jìn)先出的,其內(nèi)存空間是向上增長(zhǎng)的。
代碼示例:
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
//申請(qǐng)內(nèi)存
int *p = (int *)malloc(100);
if(p == NULL) //錯(cuò)誤檢查
{
printf("error \n");
return -1;
}
//內(nèi)存使用S
*(p+4) = 1024;
*(p+3000) = 111;
printf("*(p+3) = %d\n", *(p+4));
printf("*(p+30000)) = %d\n", *(p+3000));
printf("p = %p \n",p);
printf("(p +4) = %p \n",(p +4));
free(p); //釋放內(nèi)存
printf("*(p+3) = %d\n", *(p+4));
printf("*(p+30000)) = %d\n", *(p+3000));
printf("p = %p \n",p);
printf("(p +4) = %p \n",(p +4));
p = NULL; //避免野指針
return 0;
}
運(yùn)行結(jié)果:
*(p+3) = 1024
*(p+30000)) = 111
p = 0x8efb008
(p +4) = 0x8efb018
*(p+3) = 1024
*(p+30000)) = 111
p = 0x8efb008
(p +4) = 0x8efb018
分析:
1:堆內(nèi)存可以越界訪問(wèn),但是實(shí)際中最好還是不要,因?yàn)槟阍谑褂玫臅r(shí)候越界訪問(wèn)就意味著踩到別人了
2:堆內(nèi)存在釋放之后還可以訪問(wèn),并且訪問(wèn)的值還是不變的,說(shuō)明堆內(nèi)存也是臟的,堆內(nèi)存釋放的時(shí)候并沒(méi)有對(duì)使用過(guò)的沒(méi)存進(jìn)行清理。
三:數(shù)據(jù)段(.data )
1:一個(gè)程序主要有數(shù)據(jù)段(.data) 代碼段 和bss段,不同的段具有不同的段屬性
數(shù)據(jù)段:(又叫數(shù)據(jù)區(qū)、靜態(tài)數(shù)據(jù)區(qū)、靜態(tài)區(qū))數(shù)據(jù)段存放的是程序的中顯示初始化的全局變量(不包括初始化為0的全局變量),同時(shí)需要注意的是全局變量才算是程序的數(shù)據(jù),局部變量不是程序變量,只是函數(shù)數(shù)據(jù)
代碼段:代碼段就是程序中的可執(zhí)行部分,直觀理解代碼段就是函數(shù)堆疊組成的。代碼段是只讀的。
bss段:(又叫zero initial 段)bss段存放的是為顯示初始化的全局變量已經(jīng)初始化為0的全局變量(C語(yǔ)言中默認(rèn)全局變量的初始化就是為0),所以可以理解為bss段就是初始化為0的數(shù)據(jù)段。
2:放在.data段的變量有兩種:第一種是顯式初始化為非0的全局變量,另一種是靜態(tài)局部變量,也就是static修飾的局部變量(普通變量是分配在棧上面,靜態(tài)局部變量是分布在.data段)
四:代碼段中的特殊數(shù)據(jù)
1:C語(yǔ)言使用char *p = "linux";定義字符串的時(shí)候,字符串"linux"實(shí)際上是被分配到了代碼段上面,換句話說(shuō)這里的字符串"linux"實(shí)際上是一個(gè)常量字符串而不是變量字符串。
2:const修飾的類(lèi)型常量,const的實(shí)現(xiàn)方法主要有兩種,一種是編譯的時(shí)候?qū)onst修飾的變量放在代碼段去實(shí)現(xiàn)不可修改(因?yàn)榇a段是只讀的),另一種是由編譯器來(lái)檢查來(lái)確保const修飾的常量,但是這種實(shí)際上和普通常量樣,也是被放在了數(shù)據(jù)段,所以如果繞過(guò)編譯器就可以實(shí)現(xiàn)修改,具體實(shí)現(xiàn)方法是用指針去指向這個(gè)常量的內(nèi)存空間,然后解引用去修改這個(gè)常量,gcc中就是這樣實(shí)現(xiàn)的。
另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務(wù)器15元起步,三天無(wú)理由+7*72小時(shí)售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國(guó)服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡(jiǎn)單易用、服務(wù)可用性高、性價(jià)比高”等特點(diǎn)與優(yōu)勢(shì),專(zhuān)為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場(chǎng)景需求。
網(wǎng)頁(yè)名稱:C語(yǔ)言之程序中內(nèi)存的來(lái)源:棧堆數(shù)據(jù)段-創(chuàng)新互聯(lián)
當(dāng)前URL:http://aaarwkj.com/article26/dddecg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供建站公司、微信小程序、關(guān)鍵詞優(yōu)化、服務(wù)器托管、動(dòng)態(tài)網(wǎng)站、網(wǎng)站維護(hù)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)
猜你還喜歡下面的內(nèi)容