Android NDK開(kāi)發(fā)(C語(yǔ)言--動(dòng)態(tài)內(nèi)存分配)
1.C 內(nèi)存管理函數(shù)
C 語(yǔ)言為內(nèi)存的分配和管理提供了幾個(gè)函數(shù)。這些函數(shù)可以在 <stdlib.h>
頭文件中找到。
序號(hào) | 函數(shù)和描述 |
---|---|
1 | void calloc(int num, int size); 在內(nèi)存中動(dòng)態(tài)地分配 num 個(gè)長(zhǎng)度為 size 的連續(xù)空間,并將每一個(gè)字節(jié)都初始化為 0。所以它的結(jié)果是分配了 numsize 個(gè)字節(jié)長(zhǎng)度的內(nèi)存空間,并且每個(gè)字節(jié)的值都是0。 |
2 | void free(void *address); 該函數(shù)釋放 address 所指向的內(nèi)存塊,釋放的是動(dòng)態(tài)分配的內(nèi)存空間。 |
3 | void *malloc(int num); 在堆區(qū)分配一塊指定大小的內(nèi)存空間,用來(lái)存放數(shù)據(jù)。這塊內(nèi)存空間在函數(shù)執(zhí)行完成后不會(huì)被初始化,它們的值是未知的。 |
4 | void *realloc(void *address, int newsize); 該函數(shù)重新分配內(nèi)存,把內(nèi)存擴(kuò)展到 newsize。 |
2.C語(yǔ)音里面的內(nèi)存劃分
- 棧區(qū)(棧內(nèi)存,存放局部變量,自動(dòng)分配和釋放,里面函數(shù)的參數(shù),方法里面的臨時(shí)變量)
- 堆區(qū)(動(dòng)態(tài)內(nèi)存分配,C語(yǔ)音里面由程序員手動(dòng)分配),最大值為操作系統(tǒng)的80%
- 全局區(qū)或靜態(tài)區(qū)
- 常量區(qū)(字符串)
- 程序代碼區(qū)
3.靜態(tài)與動(dòng)態(tài)內(nèi)存分配
在程序運(yùn)行過(guò)程中,動(dòng)態(tài)指定需要使用的內(nèi)存大小,手動(dòng)釋放,釋放之后這些內(nèi)存還可以被重新使用。
靜態(tài)內(nèi)存分配,分配內(nèi)存大小的是固定,產(chǎn)生的問(wèn)題:
- 1.很容易超出棧內(nèi)存的最大值
- 2.為了防止內(nèi)存不夠用會(huì)開(kāi)辟更多的內(nèi)存,容易浪費(fèi)內(nèi)存。
動(dòng)態(tài)內(nèi)存分配,在程序運(yùn)行過(guò)程中,動(dòng)態(tài)指定需要使用的內(nèi)存大小,手動(dòng)釋放,釋放之后這些內(nèi)存還可以被重新使用
4.棧溢出
下面的代碼會(huì)導(dǎo)致棧溢出:
void main(){ //屬于靜態(tài)內(nèi)存分配,分配到棧里面,Window里面每一個(gè)應(yīng)用棧大概是2M,大小確定。與操作系統(tǒng)有關(guān)。 int a [1024 * 1024 * 10 * 4]; }
該靜態(tài)內(nèi)存定義為40M,而Window
里面每一個(gè)應(yīng)用棧大概是2M,超出了范圍, 會(huì)報(bào)stack overflow
錯(cuò)誤 。
5.動(dòng)態(tài)內(nèi)存分配與釋放
//堆存分配,40M //參數(shù):字節(jié) KB M 10M 40M //開(kāi)辟 int* p1 = (int*)malloc(1024*1024*10*sizeof(int)); //釋放 free(p1);
通過(guò)動(dòng)態(tài)內(nèi)存分配來(lái)動(dòng)態(tài)指定數(shù)組的大?。?/strong>
在程序運(yùn)行過(guò)長(zhǎng)中,可以隨意的開(kāi)辟指定大小的內(nèi)存,以供使用,相當(dāng)于Java
中的集合
#define _CRT_SECURE_NO_WARNINGS #include <stdio.h> #include <stdlib.h> #include <Windows.h> //創(chuàng)建一個(gè)數(shù)組,動(dòng)態(tài)指定數(shù)組的大小 void main() { //靜態(tài)內(nèi)存分配創(chuàng)建數(shù)組,數(shù)組的大小是固定的 //int i = 10; //int a[i]; int len; printf("輸入數(shù)組的長(zhǎng)度:"); scanf("%d", &len); //開(kāi)辟內(nèi)存,大小內(nèi)存len * 4 字節(jié) int* p = (int*)malloc(len * sizeof(int));//p:數(shù)組的首地址 int i = 0; for (; i < len; i++) { p[i] = rand() % 100; printf("%d,%#x\n", p[i], &p[i]); } //手動(dòng)釋放內(nèi)存 free(p); getchar(); }
結(jié)果輸出:
41,0x513f48
67,0x513f4c
34,0x513f50
0,0x513f54
69,0x513f58
24,0x513f5c
6.重新分配realloc
重新分配內(nèi)存的兩種情況:
縮小內(nèi)存,縮小的那一部分?jǐn)?shù)據(jù)會(huì)丟失
擴(kuò)大內(nèi)存,(連續(xù)的)
1.如果當(dāng)前內(nèi)存段后面有需要的內(nèi)存空間,直接擴(kuò)展這段內(nèi)存空間,realloc
返回原指針
2.如果當(dāng)前內(nèi)存段后面的空閑字節(jié)不夠,那么就使用堆中的第一個(gè)能夠滿足這一要求的內(nèi)存塊,將目前的數(shù)據(jù)復(fù)制到新的位置,并將原來(lái)的數(shù)據(jù)庫(kù)釋放掉,返回新的內(nèi)存地址
3.如果申請(qǐng)失敗,返回NULL,原來(lái)的指針仍然有效
void main() { int len; printf("第一次輸入數(shù)組的長(zhǎng)度:"); scanf("%d", &len); //開(kāi)辟內(nèi)存,大小內(nèi)存len * 4 字節(jié) int* p = (int*)malloc(len * sizeof(int));//p:數(shù)組的首地址 int i = 0; for (; i < len; i++) { p[i] = rand() % 100; printf("%d,%#x\n", p[i], &p[i]); } int addLen; printf("輸入數(shù)組增加的長(zhǎng)度:"); scanf("%d", &addLen); int* p2 = (int*)realloc(p, sizeof(int) * (len + addLen)); if (p2 == NULL) { printf("重新分配失敗......"); } printf("------------新數(shù)組-------------------\n"); //重新賦值 i = 0; for (; i < len + addLen; i++) { p2[i] = rand() % 200; printf("%d,%#x\n", p2[i], &p2[i]); } //手動(dòng)釋放內(nèi)存 p2釋放內(nèi)存 p也會(huì)釋放,因?yàn)榻op2分配內(nèi)存的時(shí)候要么p已經(jīng)釋放,要么p2、p指向統(tǒng)一地址區(qū)域 if (p2 != NULL) { free(p2); p2 = NULL; } getchar(); }
結(jié)果輸出:
第一次輸入數(shù)組的長(zhǎng)度:5
41,0x5e4ad8
67,0x5e4adc
34,0x5e4ae0
0,0x5e4ae4
69,0x5e4ae8
輸入數(shù)組增加的長(zhǎng)度:5
------------新數(shù)組-------------------
124,0x5e4ad8
78,0x5e4adc
158,0x5e4ae0
162,0x5e4ae4
64,0x5e4ae8
105,0x5e4aec
145,0x5e4af0
81,0x5e4af4
27,0x5e4af8
161,0x5e4afc
內(nèi)存分配的幾個(gè)注意細(xì)節(jié):
- 1.不能多次釋放
- 2.釋放完之后(指針仍然有值),給指針置NULL,標(biāo)志釋放完成
- 3.內(nèi)存泄露(p重新賦值之后,再free,并沒(méi)有真正釋放內(nèi)存)
避免內(nèi)存泄漏:
p重新賦值之前先free
內(nèi)存泄漏寫法:
void main(){ //40M int* p1 = malloc(1024 * 1024 * 10 * sizeof(int)); //free(p1); //p1 = NULL; //錯(cuò)誤,沒(méi)有立即釋放內(nèi)存 printf("%#x\n",p1); //80M p1 = malloc(1024 * 1024 * 10 * sizeof(int) * 2); free(p1); p1 = NULL; getchar(); }
打開(kāi)任務(wù)管理器,看到有40M內(nèi)存泄漏。
正確寫法:
void main(){ //40M int* p1 = malloc(1024 * 1024 * 10 * sizeof(int)); free(p1); p1 = NULL; printf("%#x\n",p1); //80M p1 = malloc(1024 * 1024 * 10 * sizeof(int) * 2); free(p1); p1 = NULL; getchar(); }
打開(kāi)任務(wù)管理器,看到內(nèi)存只有0.3M,正常。
到此這篇關(guān)于Android NDK開(kāi)發(fā)(C語(yǔ)言--動(dòng)態(tài)內(nèi)存分配)的文章就介紹到這了,更多相關(guān)Android NDK C語(yǔ)言動(dòng)態(tài)內(nèi)存分配內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Android如何使用GestureDetector進(jìn)行手勢(shì)檢測(cè)詳解
GestureDetector使用很方便,提供了單擊,雙擊,長(zhǎng)按等操作的處理,但是一般的定義界面都比較復(fù)雜,還用很多需要注意的地方,這篇文章主要給大家介紹了關(guān)于Android如何使用GestureDetector進(jìn)行手勢(shì)檢測(cè)的相關(guān)資料,需要的朋友可以參考下2022-01-01Android中Activity和Fragment傳遞數(shù)據(jù)的兩種方式
本篇文章主要介紹了Android中Activity和Fragment傳遞數(shù)據(jù)的兩種方式,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-09-09Android倒計(jì)時(shí)控件 Splash界面5秒自動(dòng)跳轉(zhuǎn)
這篇文章主要為大家詳細(xì)介紹了Android倒計(jì)時(shí)控件,Splash界面5秒自動(dòng)跳轉(zhuǎn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-09-09Android開(kāi)發(fā)筆記之:AsyncTask的應(yīng)用詳解
本篇文章是對(duì)Android中AsyncTask的應(yīng)用進(jìn)行了詳細(xì)的分析介紹,需要的朋友參考下2013-05-05Android圖片翻轉(zhuǎn)動(dòng)畫簡(jiǎn)易實(shí)現(xiàn)代碼
Android圖片翻轉(zhuǎn)動(dòng)畫效果如何實(shí)現(xiàn),本文將給你一個(gè)驚喜,實(shí)現(xiàn)代碼已經(jīng)列出,需要的朋友可以參考下2012-11-11Android Animation之TranslateAnimation(平移動(dòng)畫)
這篇文章主要為大家詳細(xì)介紹了Animation之TranslateAnimation平移動(dòng)畫,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09詳解Android 8.0以上系統(tǒng)應(yīng)用如何?;?/a>
這篇文章主要介紹了詳解Android 8.0以上系統(tǒng)應(yīng)用如何?;?,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08Android自定義View——扇形統(tǒng)計(jì)圖的實(shí)現(xiàn)代碼
本篇文章主要介紹了Android自定義View——扇形統(tǒng)計(jì)圖的實(shí)現(xiàn)代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02Android中數(shù)據(jù)庫(kù)常見(jiàn)操作實(shí)例分析
這篇文章主要介紹了Android中數(shù)據(jù)庫(kù)常見(jiàn)操作,以完整實(shí)例形式較為詳細(xì)的分析了Android基于數(shù)據(jù)庫(kù)類進(jìn)行數(shù)據(jù)庫(kù)的增刪改查等操作技巧,需要的朋友可以參考下2015-10-10