日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學無先后,達者為師

網站首頁 編程語言 正文

C語言動態開辟內存詳解_C 語言

作者:XATU弟中之弟 ? 更新時間: 2022-04-28 編程語言

1.動態內存管理

我們已經掌握的內存開辟方法,用的最多的就是數組,但是首先我們知道數組實在棧上開辟空間的,要是我們開辟大量的空間怎么辦呢?而且我不確定數據的大小萬一造成內存資源浪費是不是也不劃算呢?所以我認為一下兩點就很充分的說明動態開辟內存存在的必要性。

1.一般方式(棧上開辟)只能自動開辟少量的空間,但是堆上可以開辟大量的空間。

2.對于不定長數據保存問題,動態開辟空間可以解決。

2.動態開辟內存函數的介紹

2.1 malloc和free函數

C語言提供了一個動態開辟內存的函數。

void* malloc (size_t size);

C語言還提供了一個函數free是對動態開辟內存釋放和回收的。

void free (void* ptr);

這兩個函數都聲明在stdlib.h這個頭文件中

那如何進行內存的開辟與釋放呢?舉個例子。

int main() {
	int* p = (int*)malloc(10 * sizeof(int));
	if (NULL == p) {
		return 1;
	}
	free(p);
}

1.malloc 等空間申請都是在堆上進行申請,最后必須由free來進行釋放。堆上的空間是由程序員自己管理。

2.malloc是一個函數,表明了堆空間說在程序運行起來之后,再在系統上申請的,空間只申請不釋放,會造成內存泄露問題!

3.那free是做了什么呢?他是把開辟的空間給清除了?還是把指針給清空了?

其實都不是,free做的是取消了指針和所對應內存的指向 “關系”。

在實際申請空間的時候,真實給你的空間是要大于你所需要的,但是你只能使用你要的大小,多出來的字節,用來維護剛剛說的那種關系,以及保存該次申請的 元數據(屬性數據):用戶申請的空間有多大,所以在free傳參的時候只用傳入你開辟空間的起始地址就好了,根據屬性數據free函數就知道該釋放多少空間。

4.那我不想釋放那么多可以嗎?我按照以下代碼free。

free(p+4);

是不行的!堆空間必須整體申請整體釋放。

2.2 calloc函數

C語言還提供了一個函數calloc

void* calloc(size_t num, size_t size );

?calloc跟malloc使用基本一樣

只是有一點區別,malloc沒做初始化,隨機值,malloc效率更高一點。calloc做了初始化,效率更低一點。

2.3 relloc函數

C語言提供的這個函數讓動態內存管理更加的靈活,有時候發現申請的空間太小了,有時候覺得申請的空間太大了,合理調整內存就有了relloc函數。

void* relloc (void* ptr,size_t size);

ptr是調整內存的地址,size是調整后的大小,返回值是調整之后內存的起始地址。

一般relloc在調整內存是存在兩種情況

1.原有空間后面有足夠大的空間

直接向后擴充就好了

2.原有空間后面沒有足夠大的空間

所以說ptr也就是堆空間的起始地址有可能是變化的!

最后在分享一個題

void GetMemory(char* p)
{
	p = (char*)malloc(100);
}
int main ()
{
    char* str = NULL;
	GetMemory(str);
	strcpy(str, "hello world");
	printf(str);
}

?這個打印的是啥呢?

其實是有錯誤的 str傳入函數發生臨時拷貝問題此時p和str不是一個東西進行動態內存開辟讓我們的p指向開辟的空間,調用函數開辟棧幀,調用完畢釋放棧幀,p是一個臨時變量于那個空間已經沒有指向關系了,而str依舊是NULL?

更改如下

void GetMemory(char** p)
{
	*p = (char*)malloc(100);
}
int main()
{
	char* str = NULL;
	GetMemory(&str);
	strcpy(str, "hello world");
	printf(str);
}

總結

原文鏈接:https://blog.csdn.net/qq_50119037/article/details/123097619

欄目分類
最近更新