C語言中手把手教你動態(tài)內(nèi)存分配

C語言中手把手教你動態(tài)內(nèi)存分配

動態(tài)內(nèi)存分配

常見的內(nèi)存分配的錯誤

先上一個內(nèi)存分配的思維導(dǎo)圖:便于聯(lián)想想象,理解:

首先我們介紹一下內(nèi)存分配的方式:

1:在靜態(tài)存儲區(qū)域中進(jìn)行分配

? ? ? ? 內(nèi)存在程序編譯的時候就已經(jīng)分配好,這塊內(nèi)存在程序的整個運行期間都存在。例如全局變量,static變量


2:在棧中進(jìn)行分配

? ? ? ? 在執(zhí)行函數(shù)時,函數(shù)內(nèi)局部變量的存儲單元都可以在棧上創(chuàng)建,函數(shù)執(zhí)行結(jié)束時,這些存儲但愿自動被釋放。效率很高,但是分配的內(nèi)存容量比較有限


3:在堆中進(jìn)行分配

? ? ? ? 在堆上分配也稱為動態(tài)內(nèi)存分配:程序在運行的時候用malloc等函數(shù)申請任意多少的內(nèi)存,程序員自己負(fù)責(zé)在何時用free釋放內(nèi)存。動態(tài)內(nèi)存分配的生存期由我們自己決定,使用非常靈活,但是問題相對也比較多;注意://如果沒有釋放的話,很容易就會造成內(nèi)存溢出,因為堆中的內(nèi)存塊是全局的,因此不會因為函數(shù)的調(diào)用而結(jié)束


動態(tài)內(nèi)存分配中使用的函數(shù):

1:malloc函數(shù):需要用到的頭文件malloc.h

void *malloc(size_t size) //————–>返回的是一個通用類型的指針,根據(jù)需要去進(jìn)行強(qiáng)轉(zhuǎn);

功能:允許從空閑內(nèi)存池中分配連續(xù)內(nèi)存但不初始化

參數(shù):size參數(shù)實際就是一個所需字節(jié)數(shù)的整數(shù) malloc(20);

返回:若分配成功則返回一個指向該內(nèi)存塊的指針,在使用時可根據(jù)需要做強(qiáng)制類型轉(zhuǎn)換,否則返回NULL(空指針)//需要判空

free(p);//釋放內(nèi)存空間,將內(nèi)存釋放出來給系統(tǒng);

free函數(shù)與malloc函數(shù)是成對出現(xiàn)的;

申請malloc的時候盡量去給它進(jìn)行一下初始化,防止后面出現(xiàn)一些不確定性的東西;

malloc的生命周期:只要沒有調(diào)用free這個函數(shù),進(jìn)程沒有結(jié)束,那么此時,這個函數(shù)的生命周期就會一直存在在內(nèi)存中;它是存放在堆空間中的,它不會因為你去函數(shù)調(diào)用的結(jié)束自動去釋放,堆當(dāng)中的內(nèi)存是全局的

如:int p = (int )malloc(n*sizeof(int)); //在空閑內(nèi)存池中分配連續(xù)內(nèi)存n*sizeof(int)個字節(jié)的堆內(nèi)存空間

malloc的相關(guān)實例代碼如下:

#include<stdio.h>

#include<malloc.h>

void out(int *p,int n)

{

? ? int i;

? ? for(i=0;i<n;i++)

? ? {?

? ? ? ? printf("%d",*(p+i));

? ? ? ? printf("---------------\n");

? ? }?

}

int main(void)

{

? ? printf("please input one number:");

? ? int n;

? ? scanf("%d",&n);

? ? //申請

? ? int *p = (int *)malloc(n * sizeof(int));

? ? //內(nèi)存申請成功

? ? if(p != NULL){

? ? ? ? out(p,n);

? ? ? ? int i;

? ? ? ? for(i=0;i<n;i++){

? ? ? ? ? ? *(p+i)=i*i;

? ? ? ? }

? ? ? ? out(p,n);

? ? ? ? //釋放掉堆內(nèi)存

? ? ? ? free(p);

? ? }else{

? ? ? ? //內(nèi)存申請失敗

? ? ? ? printf("malloc is NULL!\n");

? ? }?

? ? return 0;

}

2:calloc函數(shù):需要用到的頭文件stdlib.h

void *colloc(size_t num_elements,size_t element_size);

功能:功能同malloc是一樣的,但是作初始化

參數(shù):num_elements是所需的元素的數(shù)量,element_size是每個元素的字節(jié)數(shù)

返回:同malloc函數(shù)一樣

也是需要與free(p)進(jìn)行對稱使用

calloc相關(guān)代碼如下所示:

#include<stdio.h>

#include<stdlib.h>

int main(void)

{

? ? printf("please input one number:");

? ? int n;

? ? scanf("%d",&n);

? ? int *p = (int *)calloc(n,sizeof(int));

? ? if(p!=NULL){

? ? ? ? int i;

? ? ? ? for(i=0;i<n;i++)

? ? ? ? {

? ? ? ? ? ? printf("%d ",*(p+i));

? ? ? ? }

? ? ? ? printf("\n");

? ? ? ? free(p);

? ? }else{

? ? ? ? printf("calloc error\n");

? ? }?

? ? return 0;

}

3: realloc函數(shù):需要用到的頭文件(stdlib.h),動態(tài)擴(kuò)大縮小申請的內(nèi)存

void *realloc(void *ptr,size_t new_size);

功能:在指針ptr指向的內(nèi)存基礎(chǔ)上擴(kuò)大或者縮小內(nèi)存

參數(shù):ptr是指向先前通過malloc,calloc和realloc函數(shù)后分配的內(nèi)存塊的指針,new_size是內(nèi)存塊的新尺寸,可能大于或者小于原有內(nèi)存尺寸;這個是追加到new_size的新的內(nèi)存

realloc在C語言中也被稱為動態(tài)數(shù)組;

realloc函數(shù)使用的注意點:

1:當(dāng)擴(kuò)展內(nèi)存的時候,不會對添加進(jìn)內(nèi)存塊的字節(jié)進(jìn)行初始化

2:若不能調(diào)整內(nèi)存則返回NULL,但原有內(nèi)存中的數(shù)據(jù)是不會發(fā)生改變的

3:若第一個參數(shù)為NULL那么功能 等同與malloc函數(shù),若第二個參數(shù)為0,那么會釋放調(diào)用內(nèi)存塊

realloc(NULL,10*size(int)) 等同malloc(10*sizeof(int));

realloc(p,0); 等同于free

4:當(dāng)縮小或者擴(kuò)大內(nèi)存時,一般不會對其進(jìn)行移動,若無法擴(kuò)大內(nèi)存塊,那么啃呢個會在別處分配新的內(nèi)存快,然后把舊內(nèi)存塊的數(shù)據(jù)復(fù)制到新塊 中,并將舊塊刪除釋放內(nèi)存;

realloc相關(guān)的的代碼為:

#include<stdlib.h>

#include<stdio.h>

#include<malloc.h>

void out(int *p ,int n){

? ? int i;

? ? for(i = 0 ;i < n; i++){

? ? ? ? printf("%d\n",*(p+i));

? ? }?

}

int main(void)

{?

? ? //申請4個字節(jié)的堆內(nèi)存空間,未初始化

? ? int * p = (int *)malloc(5*sizeof(int));

? ? if(p == NULL) exit(1);

? ? *p = 1;

? ? *(p+1)? =2;

? ? p[2] = 3;

? ? p[3] = 4;

? ? p[4] = 5;

? ? out(p,5);

? ? printf("===============\n");

? ? //追加申請10個字節(jié)的內(nèi)存空間,追加的空間也是未進(jìn)行初始化的

? ? p = (int *)realloc(p,10*sizeof(int));

? ? if(p == NULL) exit(1);

? ? p[6] = 6;

? ? *(p+6) = 7;

? ? *(p+7) = 8;

? ? *(p+8) = 9;

? ? *(p+9) = 10;

? ? out(p,10);

? ? free(p);

? ? //free之后,將指針置為空

? ? p = NULL;

? ? return 0;

}

4:free函數(shù)

free之后如果還有這塊內(nèi)存地址的話,此時這塊內(nèi)存歸還給了系統(tǒng),(可能這塊內(nèi)存還處于一個空閑狀態(tài))但是還是可以對其進(jìn)行操作。里面的值短暫的會保留。

free之后,申請內(nèi)存的那個指針就會變成野指針(聲明了,但是沒有任何指向的指針),有時候會出現(xiàn)野指針錯誤;

所以盡量在操作之后:將指針置為NULL

p=NULL;

注意:申請和釋放是成對的,所以程序是不能進(jìn)行多次free的,否則會崩潰的

常見的內(nèi)存錯誤:

1:段錯誤

使用未分配成功的內(nèi)存

避免方式:在使用內(nèi)存之前檢查指針是否為NULL;

引用分配成功但尚未初始化的內(nèi)存

避免方式:賦予初值,即便是賦予零值也不可省略

內(nèi)存分配成功并且已經(jīng)初始化,但操作越過了內(nèi)存的邊界

避免:注意下表的使用不能超出邊界

忘記釋放內(nèi)存,造成內(nèi)存泄露

避免方式:申請內(nèi)存的方式和釋放內(nèi)存的方式需要成雙成對

釋放內(nèi)存之后卻繼續(xù)去使用這一塊內(nèi)存

避免方式:使用free內(nèi)存之后,把指針置為NULL;

內(nèi)存錯誤的注意點:

指針消亡了,并不表示它所指向的內(nèi)存會被自動釋放,(在free之前,直接將指針設(shè)為NULL);

內(nèi)存釋放了,并不代表指針會消亡或者成了NULL指針;(在free之后,指針并沒有進(jìn)行NULL設(shè)置)

野指針:

野指針的形成是指針變量沒有被初始化,任何指針變量剛被創(chuàng)建的時候不會自動成為NULL指針,它的缺省值是最忌的,它會亂指一氣

指針變量在創(chuàng)建的同時應(yīng)當(dāng)被初始化,要么將指針設(shè)置為NULL,要么讓它指向合法內(nèi)存

free內(nèi)存塊之后,需要將指針設(shè)置為NULL,如果沒有設(shè)置為NULL,也會出現(xiàn)“野指針”,它是指向“垃圾”內(nèi)存的指針;

多次free內(nèi)存塊,是會導(dǎo)致程序崩潰的

重要的事說三遍~~~~

交流728483370,一起學(xué)習(xí)加油!

交流728483370,一起學(xué)習(xí)加油!

交流728483370,一起學(xué)習(xí)加油!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容