當前位置:首頁 » 編程語言 » c語言動態分配空間

c語言動態分配空間

發布時間: 2024-10-09 03:35:14

c語言-動態分配內存 malloc & free

需要用一個數組來保存用戶的輸入,但是卻不知道用戶會輸入多少條數據。

(1) 如果設一個太大的數組,則顯得浪費內存
(2) 如果設得太小,又怕不夠

問題:如何做到恰好夠用、又一點不浪費呢?

系統中存在一個內存管理器(MM, Memory Manager),它負責管理一堆閑置內存。它被設計用於解決此類問題。

MM提供的服務:應用程序可以向MM申請(借出)一塊指定大小的內存,用完之後再釋放(還回)。

應用程序在使用malloc時,要把返回值轉換成目標類型。

這塊內存和數組沒有本質區別,用法完全相同。

需要先計算需要多少位元組的內存空間

數組舉例子:

釋放的時候需要注意, 因為在for循環執行之後,p的地址往前移動了10, 所以需要減去10, 然後再釋放p,不然會有問題

// 當銷毀時只需要free一次,malloc了幾個位元組就會free幾個位元組,和char類型還是int類型無關
free(p);

在一個函數中動態分配的內存,在另一個函數中操作這塊內存

(1) MM是一個系統級的東西,所有的應用程序都向同一個MM申請內存。

(2) 何為借出?實際上,在內存被借出時,MM只是把它管理的內存標記了一下,表示該段內存已經被佔用。比如,它把每一段被佔用的內存給記錄下來(首地址,長度)
(p0,n0) (p1, n1) (p2, n2) ...

(3) MM非常慷慨:①只要有人 malloc ,它都同意借出 ②你不歸還,它永遠不會主動要求你 free 。

(4) MM管理的內存區域稱為「堆」Heap

這意味著,用戶程序應該自覺得及時 free ,以便不耽誤別的應用程序的使用。如果有個應用程序不停地 malloc ,而不 free ,那最終會用光MM的內存。當MM沒有更多閑置內存時, malloc 返回 NULL ,表示內存已經用完。

再次重申: 應用程序在malloc之後,應該盡早free !

使用原則:需要的時候再申請,不需要的時候立即釋放

實際上,MM對借出的內存塊進行標識
(p0, n0) (p1, n1) (p2, n2) ...
它內部已經保證任意兩塊內存不會「交疊」,即不會重疊,不會把一塊內存同時借給兩個應用程序使用。

所以,每塊內存的首地址都是不同的,在 free 的時候只需要指明首地址即可。

對象指的一塊內存

示例:用Citizen表示一個市民,用Car表示一個輛車。他起初沒有車,但未來可能有一輛車。

怎麼樣才算「及時」? 「不及時」會怎樣?

MM里可用的內存是有限的,你用完了就得盡快還,因為別的應用程序也需要MM的內存。

只借不還,積累到一定程度,MM沒有更多內存可用,於是malloc返回NULL。

要還就得全還,否則MM那邊處理不了

原因是:MM可能此時沒有閑置內存可用。(雖然這種情況一般不會發生)

free之後,該內存交還給MM,該內存不再可用(失效)

不一定要在相同的函數里釋放,在應用程序的任意一個角落釋放都是有效的。

也就是說:這一塊內存被malloc出來之後,完全交給你處置

功能:將 s 中當前位置後面的 n 個位元組 (typedef unsigned int size_t )用 ch 替換並返回 s

參數:

參數:

功能:由 src 所指內存區域復制 n 個位元組到 dest 所指內存區域。

memmove() 功能用法和 memcpy()) 一樣,區別在於: dest
和 src 所指的內存空間重疊時, memmove() 仍然能處理,不過執行效率比 memcpy() 低一些

② C語言動態內存函數分配問題

你的問題是剛剛學習指針的人都有的問題,需要從地址來認識這個問題,C語言的數據都是需要申請空間來存放的,有靜態和動態兩種,動態分配的空間大小可以按需要分配,並且可以回收,靜態是不可以回收的。

首先,分配動態的空間,指針P和空間的首地址是沒有直接的聯系的,只是為了手續要使用這個剛剛分配的空間才讓P指針指向這個剛剛分配的空間的首地址的話,如果還不理解,你可以反向想想如果你分配的這個空間,而不用P指針指向這個首地址,那麼你如何使用這個空間呢?
void型是針對地址型的直接分配,為什麼不是分配char的空間呢?因為C語言和硬體關系密切,
有寫硬體他的地址不是8位存儲的,是16位,或者是32位的,那麼這個void就忽略了這個硬體地址寬位就分配100個地址就是了,然後,強制轉換成char的8位,如果硬體是16位了,那麼,每個地址就會多分配了8位,而使用前8為來存儲空間了。

返回的指針是分配的100個地址的首地址,不是這個指針P,舉個例子吧,空間可以看做是一個隊伍,他本身就有一個領頭的人,那麼P就是一個單獨的人,那麼,我們如果像要找到這個隊伍,需要先找到隊伍的領頭人,領頭的人只在你申請的時候出現一次,後面,你就找不到他了,怎麼辦?你把他電話給P吧,P可以找到他。呵呵。

如果還是不理解,那就不要理解了,建議去吧隊列的方式,用指針和動態分配內存的方式實現一次,就知道了,不能照著書本抄,全部代碼要自己寫出來,就肯定理解了。

③ c語言中,malloc和free是什麼意思

屬於內存管理的兩個函數,malloc是申請內存的,free是釋放內存的。

1、malloc一般用法:

int *t=NULL;

t=(int *)malloc(sizeof(int));

也可以在sizeof前面加上一個'n*'這就成了一個動態分配數組的方法。

2、free一般用法:

int *t=NULL;

t=(int *)malloc(sizeof(int));

free(t);

這樣t所指的空間就被釋放掉了。

(3)c語言動態分配空間擴展閱讀:

malloc函數定義

其函數原型為void *malloc(unsigned int size);其作用是在內存的動態存儲區中分配一個長度為size的連續空間。此函數的返回值是分配區域的起始地址,或者說,此函數是一個指針型函數,返回的指針指向該分配域的開頭位置。

如果分配成功則返回指向被分配內存的指針(此存儲區中的初始值不確定),否則返回空指針NULL。當內存不再使用時,應使用free()函數將內存塊釋放。

④ C語言中的動態內存分配的用法舉例

1、malloc函數:其作用是在內存的動態存儲區中分配一個長度為size的連續空間。其參數是一個無符號整形數,返回值是一個指向所分配的連續存儲域的起始地址的指針。

2、free函數:由於內存區域總是有限的,不能不限制地分配下去,而且一個程序要盡量節省資源,所以當所分配的內存區域不用時,就要釋放它,以便其它的變數或者程序使用。這時我們就要用到free函數。

3、calloc函數:其作用是在內存的動態存儲區中分配n個長度為 size 的連續空間。函數返回一個指向分配區域的起始位置的指針;如果分配不成功,則返回NULL。

(4)c語言動態分配空間擴展閱讀:

函數運算符:

new

運算符new用於向系統申請動態存儲空間,並把首地址作為運算結果,它的使用形式為:

指針變數=new 數據類型;

例如:

int *p=new int

該語句的作用是會用new從內存中申請了一個int型變數(4個位元組),並將該變數的首地址賦給指針變數p。

new所建立的變數的初始值是任意的,也可在用new分配內存的同時進行初始化。使用形式為:

指針變數=new 數據類型(初始值)。

delete

堆內存可按照要求進行分配,程序對內存的需求量隨時會發生變化,有時程序在運行種可能會不再需要由new分配的內存空間,而且程序還未運行結束,這時就需要把先前佔用的內存空間釋放給堆內存,以後重新分配,供程序的其他部分使用。運算符delete用於釋放new分配的內存空間,刪除建立的對象,它的使用形式為:

delete指針變數;

其中的指針變數中保存著new分配的內存的首地址。

⑤ c語言中malloc是什麼怎麼用

malloc() 函數用來動態地分配內存空間,其原型為:void* malloc (size_t size);

說明:

【參數說明】

size 為需要分配的內存空間的大小,以位元組(Byte)計。

【函數說明】

malloc() 在堆區分配一塊指定大小的內存空間,用來存放數據。這塊內存空間在函數執行完成後不會被初始化,它們的值是未知的。如果希望在分配內存的同時進行初始化,請使用 calloc() 函數。

【返回值】

分配成功返回指向該內存的地址,失敗則返回 NULL。

操作:

由於申請內存空間時可能有也可能沒有,所以需要自行判斷是否申請成功,再進行後續操作。

如果 size 的值為 0,那麼返回值會因標准庫實現的不同而不同,可能是 NULL,也可能不是,但返回的指針不應該再次被引用。

注意:函數的返回值類型是 void *,void 並不是說沒有返回值或者返回空指針,而是返回的指針類型未知。所以在使用 malloc() 時通常需要進行強制類型轉換,將 void 指針轉換成我們希望的類型,例如:

#include<stdlib.h>

typedef int ListData;

ListData *data; //存儲空間基址

data = ( ListData * ) malloc( 100 * sizeof ( ListData ) );

(5)c語言動態分配空間擴展閱讀

實現malloc的方法:

(1)數據結構

首先我們要確定所採用的數據結構。一個簡單可行方案是將堆內存空間以塊的形式組織起來,每個塊由meta區和數據區組成,meta區記錄數據塊的元信息(數據區大小、空閑標志位、指針等等)。

數據區是真實分配的內存區域,並且數據區的第一個位元組地址即為malloc返回的地址 。

(2)尋找合適的block

現在考慮如何在block鏈中查找合適的block。一般來說有兩種查找演算法

First fit:從頭開始,使用第一個數據區大小大於要求size的塊所謂此次分配的塊

Best fit:從頭開始,遍歷所有塊,使用數據區大小大於size且差值最小的塊作為此次分配的塊

兩種方式各有千秋,best fit有較高的內存使用率(payload較高),而first fit具有較高的運行效率。這里我們採用first fit演算法。

(3)開辟新的block
如果現有block都不能滿足size的要求,則需要在鏈表最後開辟一個新的block。

(4)分裂block
First fit有一個比較致命的缺點,就是可能會讓更小的size占據很大的一塊block,此時,為了提高payload,應該在剩餘數據區足夠大的情況下,將其分裂為一個新的block。

(5)malloc的實現

有了上面的代碼,我們就可以實現一個簡單的malloc.注意首先我們要定義個block鏈表的頭first_block,初始化為NULL;另外,我們需要剩餘空間至少有BLOCK_SIZE+8才執行分裂操作

由於我們需要malloc分配的數據區是按8位元組對齊,所以size不為8的倍數時,我們需要將size調整為大於size的最小的8的倍數。



熱點內容
演算法工作原理 發布:2025-01-12 20:36:38 瀏覽:24
網路訪問監控軟體 發布:2025-01-12 20:26:57 瀏覽:465
養羊啦源碼 發布:2025-01-12 20:25:48 瀏覽:570
軒逸朗逸哪個配置最好 發布:2025-01-12 20:10:00 瀏覽:49
主板存儲器分 發布:2025-01-12 20:04:46 瀏覽:376
資料庫邏輯運算 發布:2025-01-12 20:03:54 瀏覽:571
javawindows伺服器搭建 發布:2025-01-12 19:59:37 瀏覽:570
linux關閉iptables 發布:2025-01-12 19:58:49 瀏覽:150
伺服器電腦名字改了影響資料庫嗎 發布:2025-01-12 19:58:44 瀏覽:652
手機存儲優化 發布:2025-01-12 19:58:43 瀏覽:356