Malloc ir Casting supratimas C
C programavimo metu dinaminis atminties paskirstymas dažnai valdomas naudojant „malloc“ funkciją. Kūrėjai dažnai ginčijasi, ar priskirti žymeklį „malloc“ rezultatą perduoti. Pavyzdžiui, ar geriau naudoti `int *sietas = malloc(sizeof(*sieve) * ilgis);` vietoj `int *sieve = (int *)malloc(sizeof(*sieve) * ilgis);`?
Šiame straipsnyje gilinamasi į priežastis, kodėl vengiama „malloc“ rezultatų. Išnagrinėsime abiejų požiūrių pasekmes ir paaiškinsime, kodėl vienam gali būti teikiama pirmenybė prieš kitą. Šių niuansų supratimas gali padėti pagerinti jūsų C programavimo praktiką.
komandą | apibūdinimas |
---|---|
malloc | Paskiria nurodytą atminties baitų skaičių ir grąžina žymeklį į paskirtą atmintį. |
sizeof | Nustato kintamojo arba duomenų tipo dydį baitais. |
fprintf | Spausdina suformatuotą eilutę į nurodytą srautą, pvz., stderr. |
perror | Išspausdina aprašomąjį klaidos pranešimą stderr pagal paskutinę įvykusią klaidą. |
EXIT_FAILURE | Makrokomandas, rodantis nesėkmingą programos nutraukimą. |
free | Panaikina anksčiau skirtos atminties paskirstymą. |
Giliai pasinerkite į malloc ir atminties valdymą C
Pirmajame scenarijuje matome naudojimą malloc dinamiškai paskirstyti atmintį sveikųjų skaičių masyvei. Pareiškimas int *sieve = malloc(sizeof(*sieve) * length); prašo atminties „ilgio“ sveikųjų skaičių. Naudojant sizeof(*sieve), užtikriname, kad būtų paskirstytas tinkamas atminties kiekis, neatsižvelgiant į žymeklio tipą. Šis metodas leidžia išvengti rezultato liejimo poreikio malloc. Jei atminties paskirstymas nepavyksta, programa naudoja fprintf(stderr, "Memory allocation failed\n"); Norėdami išspausdinti klaidos pranešimą į standartinį klaidų srautą, tada išeikite su ne nuline būsena. Paskirta atmintis naudojama sveikiesiems skaičiams nuo 1 iki „ilgio“ saugoti ir vėliau išspausdinama prieš atlaisvinant free(sieve);.
Antrajame scenarijuje laikomės panašios struktūros, bet vietoj to skiriame atmintį dublių masyvai. Linija double *array = malloc(sizeof(*array) * length); paskiria atmintį „ilgio“ dublių skaičiui. Jei paskirstymas nepavyksta, perror funkcija išspausdina aprašomąjį klaidos pranešimą ir programa išeina su EXIT_FAILURE. Paskirstyta atmintis naudojama dviguboms reikšmėms saugoti, kurios inicijuojamos į lyginius skaičius. Šios reikšmės išspausdinamos ir galiausiai atlaisvinama atmintis naudojant free(array);. Abu scenarijai parodo, kaip svarbu patikrinti sėkmę malloc ir tinkamą naudojimą free kad išvengtumėte atminties nutekėjimo.
Supratimas apie teisingą malloc naudojimą C
C programavimas
#include <stdio.h>
#include <stdlib.h>
int main() {
int length = 10;
int *sieve = malloc(sizeof(*sieve) * length);
if (sieve == ) {
fprintf(stderr, "Memory allocation failed\\n");
return 1;
}
for (int i = 0; i < length; i++) {
sieve[i] = i + 1;
}
for (int i = 0; i < length; i++) {
printf("%d ", sieve[i]);
}
printf("\\n");
free(sieve);
return 0;
}
Atminties paskirstymo be perdavimo tyrinėjimas C
C programavimas
#include <stdio.h>
#include <stdlib.h>
int main() {
int length = 5;
double *array = malloc(sizeof(*array) * length);
if (array == ) {
perror("Failed to allocate memory");
return EXIT_FAILURE;
}
for (int i = 0; i < length; i++) {
array[i] = i * 2.0;
}
for (int i = 0; i < length; i++) {
printf("%f\\n", array[i]);
}
free(array);
return 0;
}
Atminties paskirstymo niuansai C
Kitas svarbus atminties paskirstymo aspektas C yra skirtumų supratimas malloc ir kitos atminties paskirstymo funkcijos, pvz calloc ir realloc. Nors malloc paskirsto atminties bloką jo neinicijuodamas, calloc ir paskirsto, ir inicijuoja atminties bloką iki nulio. Tai gali užkirsti kelią tam tikrų tipų klaidoms, atsirandančioms naudojant neinicializuotą atmintį. Pavyzdžiui, int *arr = calloc(length, sizeof(*arr)); užtikrina, kad visi elementai būtų inicijuoti nuliui, o tai naudinga, kai reikia švaraus lapo.
Iš kitos pusės, realloc naudojamas esamo atminties bloko dydžiui pakeisti. Jei reikia pakeisti skirto atminties bloko dydį, realloc gali būti efektyvesnis pasirinkimas nei naujo bloko paskyrimas ir turinio kopijavimas. Pavyzdžiui, arr = realloc(arr, new_length * sizeof(*arr)); koreguoja nurodyto atminties bloko dydį arr apgyvendinti new_length elementai. Tačiau tai labai svarbu tvarkyti realloc atsargiai, kad išvengtumėte atminties nutekėjimo arba neprarastumėte pradinio atminties bloko realloc nepavyksta.
Dažni klausimai ir atsakymai apie malloc C
- Kas daro malloc ginti?
- malloc reiškia „atminties paskirstymas“.
- Kodėl turėtume tikrinti rezultatą malloc?
- Mes patikriname rezultatą malloc kad būtų užtikrintas sėkmingas atminties paskirstymas ir išvengta nulinės rodyklės panaikinimo.
- Kas atsitiks, jei malloc nepavyksta?
- Jeigu malloc nepavyksta, jis grąžina nulinę rodyklę, kurią reikia patikrinti, kad būtų išvengta neapibrėžto elgesio.
- Gali malloc grąžinti nulinę žymeklį, net jei yra pakankamai atminties?
- Taip, gali sukelti kiti veiksniai, pvz., susiskaidymas malloc žlugti.
- Koks skirtumas tarp malloc ir calloc?
- malloc skiria neinicijuotą atmintį, tuo tarpu calloc paskirsto ir inicijuoja atmintį iki nulio.
- Kaip realloc dirbti?
- realloc pakeičia esamo atminties bloko dydį, išsaugodamas turinį iki naujo dydžio arba pradinio dydžio, atsižvelgiant į tai, kuris yra mažesnis.
- Ar būtina atlaisvinti atmintį, skirtą malloc?
- Taip, nepavykus atlaisvinti atminties, atsiranda atminties nutekėjimo, o tai laikui bėgant gali išeikvoti sistemos atmintį.
Pagrindiniai „Malloc Casting“ pasiūlymai:
Apibendrinant, liejant rezultatą malloc C nėra būtinas, todėl gali būti mažiau skaitomas kodas ir galimos klaidos. Praleidę atvaizdą, laikomės C standartų ir palaikome suderinamumą su C++ kompiliatoriais. Visada patikrinkite rezultatą malloc kad būtų užtikrintas sėkmingas atminties paskirstymas, ir nepamirškite atlaisvinti skirtos atminties, kad išvengtumėte nutekėjimo. Ši praktika prisideda prie patikimesnio ir prižiūrimesnio C kodo, didinant bendrą programos stabilumą.