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ą dinamiškai paskirstyti atmintį sveikųjų skaičių masyvei. Pareiškimas prašo atminties „ilgio“ sveikųjų skaičių. Naudojant , 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 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 .
Antrajame scenarijuje laikomės panašios struktūros, bet vietoj to skiriame atmintį dublių masyvai. Linija paskiria atmintį „ilgio“ dublių skaičiui. Jei paskirstymas nepavyksta, funkcija išspausdina aprašomąjį klaidos pranešimą ir programa išeina su . 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ę ir tinkamą naudojimą 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 ir kitos atminties paskirstymo funkcijos, pvz ir . Nors malloc paskirsto atminties bloką jo neinicijuodamas, ir paskirsto, ir inicijuoja atminties bloką iki nulio. Tai gali užkirsti kelią tam tikrų tipų klaidoms, atsirandančioms naudojant neinicializuotą atmintį. Pavyzdžiui, užtikrina, kad visi elementai būtų inicijuoti nuliui, o tai naudinga, kai reikia švaraus lapo.
Iš kitos pusės, naudojamas esamo atminties bloko dydžiui pakeisti. Jei reikia pakeisti skirto atminties bloko dydį, gali būti efektyvesnis pasirinkimas nei naujo bloko paskyrimas ir turinio kopijavimas. Pavyzdžiui, koreguoja nurodyto atminties bloko dydį arr apgyvendinti elementai. Tačiau tai labai svarbu tvarkyti atsargiai, kad išvengtumėte atminties nutekėjimo arba neprarastumėte pradinio atminties bloko nepavyksta.
Dažni klausimai ir atsakymai apie malloc C
- Kas daro ginti?
- reiškia „atminties paskirstymas“.
- Kodėl turėtume tikrinti rezultatą ?
- Mes patikriname rezultatą kad būtų užtikrintas sėkmingas atminties paskirstymas ir išvengta nulinės rodyklės panaikinimo.
- Kas atsitiks, jei nepavyksta?
- Jeigu nepavyksta, jis grąžina nulinę rodyklę, kurią reikia patikrinti, kad būtų išvengta neapibrėžto elgesio.
- Gali grąžinti nulinę žymeklį, net jei yra pakankamai atminties?
- Taip, gali sukelti kiti veiksniai, pvz., susiskaidymas žlugti.
- Koks skirtumas tarp ir ?
- skiria neinicijuotą atmintį, tuo tarpu paskirsto ir inicijuoja atmintį iki nulio.
- Kaip dirbti?
- 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ą ?
- Taip, nepavykus atlaisvinti atminties, atsiranda atminties nutekėjimo, o tai laikui bėgant gali išeikvoti sistemos atmintį.
Apibendrinant, liejant rezultatą 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ą 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ą.