View Full Version : linguaggio c curiosita' su malloc()
Prince_81
09-07-2008, 16:47
Volevo chiedervi info su un dubbio ke ho nell'uso di malloc questo ke vi riporto è un esempio:
void main ()
{
short num=4;
char *testo;
testo=(char *)malloc(num);
printf("%d\n",strlen(testo));
}
l'output del programma è 14 ed è questo il mio dubbio ma io volevo creare una stringa ke contenesse 4 caratteri e me ne ritrovo una ke ne contiene 14? oppure il risultato significa qualcos'altro?
wingman87
09-07-2008, 16:54
La risposta è un po' nella malloc, un po' nella strlen. La malloc alloca la memoria richiesta senza inizializzarla, quindi dentro può esserci qualunque cosa, invece la strlen restituisce il numero di caratteri tra l'inizio della stringa (che gli passi come parametro) e il primo carattere null ('\0') trovato.
PS: quando usi malloc ricordati di fare una free dopo
Prince_81
09-07-2008, 17:00
ho apportato la modifica ke mi hai consigliato
void main ()
{
short num=4;
char *testo;
testo=(char *)malloc(num);
free(testo);
printf("%d\n",strlen(testo));
}
adesso l'output mi segna 9 potrei inserire quindi 7 caretteri ma nn è quello ke io voglio giusto?
altrimenti c'e' la calloc() (prototipo analogo) che inizializza tutto a zero-NULL
usatele meno che potete, l'abuso distrugge i programmi specie se si utilizzano compilatori malsani
Prince_81
09-07-2008, 17:36
la cosa ke non mi è kiare è ke se dopo la malloc la variabile può contenere qualsiasi cosa allora il carattere nullo dovrebbe essere entro i 4 byte quindi la strlen dovrebbe segnalarmi al massimo 4 oppure di meno ma non di più, non so se mi sono spiegato.
wingman87
09-07-2008, 20:08
la cosa ke non mi è kiare è ke se dopo la malloc la variabile può contenere qualsiasi cosa allora il carattere nullo dovrebbe essere entro i 4 byte quindi la strlen dovrebbe segnalarmi al massimo 4 oppure di meno ma non di più, non so se mi sono spiegato.
No, il carattere nullo è uno dei possibili caratteri che possono trovarsi nella stringa ma non è detto che ci sia se non ce l'hai messo esplicitamente. E lo stesso vale per l'area di memoria che si trova dopo quella che hai allocato, non si sa dove si trovi il primo carattere null.
Nel tuo esempio:
Allochi 4 byte, ma non li inizializzi cosa contengono? Chissà...
Lanci la strlen. Questa parte dal primo byte e controlla se è null, siccome non lo è (è un caso, poteva benissimo esserlo) incrementa il contatore interno e passa al carattere successivo. La stessa cosa si verifica per il secondo, il terzo e il quarto carattere, quindi strlen prosegue oltre il limite di 4 caratteri e va avanti così finché non trova il carattere null. Quando lo trova ti restituisce il valore del contatore.
DanieleC88
09-07-2008, 21:04
#include <stdlib.h>
#include <string.h>
int main()
{
char *stringa;
stringa = (char *) malloc(5 * sizeof(char)); /* 4 caratteri + il terminatore '\0' */
memset(stringa, 0, 5 * sizeof(char));
strcpy(stringa, "ciao");
printf("Ho ottenuto: \"%s\".\n", stringa);
free(stringa);
return 0;
}
è bene ricordarsi anche di fare il controllo sulla riuscita della malloc onde evitare casini e puntatori a caiser
malloc restituisce null in caso di errore, meglio aprirla (sempre in cast come sopra) incapsulata in un if con controllo sul == NULL ;)
Prince_81
10-07-2008, 13:37
Grazie wingman adesso le cose mi sono più chiare non avevo pensato a memset per inizializzare le variabili :doh: , peccato che malloc non inizializzi le variabili ma forse per questo hanno pensato a memset.
Prince_81
10-07-2008, 13:39
Stev hai ragione è meglio fare sempre dei controlli con queste funzioni onde evitare casini.
Grazie wingman adesso le cose mi sono più chiare non avevo pensato a memset per inizializzare le variabili :doh: , peccato che malloc non inizializzi le variabili ma forse per questo hanno pensato a memset.
c'e' la calloc
alloca a NULL n-byte
stdecden
11-07-2008, 11:56
la cosa ke non mi è kiare è ke se dopo la malloc la variabile può contenere qualsiasi cosa allora il carattere nullo dovrebbe essere entro i 4 byte quindi la strlen dovrebbe segnalarmi al massimo 4 oppure di meno ma non di più, non so se mi sono spiegato.
Comunque se vuoi uno spazio di memoria che possa contenere stringhe fino a 4 caratteri, dovrebbe essere di 5 byte (4 byte per i caratteri e il "null terminator")
c'e' la calloc
alloca a NULL n-byte
Sì ma in pratica non c'entra niente con la richiesta dell'utente, che è solo un problema di strlen: usando calloc addirittura strlen restituirebbe 0.
Il problema è nel come è posta la domanda: tu non hai creato una "stringa di 4 caratteri", ma hai allocato 4 byte e hai dunque fatto puntare un char* a quell'area di memoria.
Se ci copi, con strcpy, una stringa di 2 caratteri, strlen ti restituirà 2, se ce ne copi una di 10 strlen restituirà 10, ma questo non ti salverà dal casino di essere andato "oltre" lo spazio che avevi riservato, magari sovrascrivendo altre variabili.
Nel tuo caso, per un puro caso 15 byte oltre il puntatore c'è una locazione di memoria che contiene uno 0, dunque il risultato di strlen è 14.
Ciò non toglie che usare strlen su un char* che punta a memoria "vergine" e non a una stringa opportunamente terminata restituisce un valore che non ha alcun senso ;)
si'
pero' siccome mi pare lui sia qui per imparare non credo gli faccia schifo sapere che se gli serve esiste un prototipo di funzione che gli inizializza a zero tutta l'area di memoria allocata e che in ottica futura puo' metterlo al riparo da errori o effetti collaterali causati da funzioni insicure che non tengono conto del finestringa (sprintf, ecc...)
si'
pero' siccome mi pare lui sia qui per imparare non credo gli faccia schifo sapere che se gli serve esiste un prototipo di funzione che gli inizializza a zero tutta l'area di memoria allocata e che in ottica futura puo' metterlo al riparo da errori o effetti collaterali causati da funzioni insicure che non tengono conto del finestringa (sprintf, ecc...)
Certo, era per non confondergli le idee.
Purtroppo, proprio perchè è un principiante, è terribile che non possa ragionare secondo le operazioni che dovrebbero banalmente essere definite su questi tipi ma debba farlo in ottica di puntatori.
Quello che intendevo dire è che anche inizializzando quell'area di memoria, non avrebbe comunque ottenuto i risultati che si aspettava dalla strlen ;)
DanieleC88
12-07-2008, 00:30
Purtroppo, proprio perchè è un principiante, è terribile che non possa ragionare secondo le operazioni che dovrebbero banalmente essere definite su questi tipi ma debba farlo in ottica di puntatori.
Come non quotarti... Venendo da linguaggi di più "alto livello" del C è stato il primo grosso ostacolo da superare... E comunque avevo già un po' di "esperienza", non oso immaginare chi inizia a programmare pensando a certa roba... brrr.
che roba ?
adesso ci sono sti dialetti... c#... porcherie varie....
DanieleC88
12-07-2008, 01:19
Ad allocare, gestire, e deallocare correttamente la memoria anche se devi fare cose banali come leggere una sola parola da un file (che non sai a priori quanto sarà lunga) o concatenare due stringhette. Con altri linguaggi basta "stampare Stringa1 + Stringa2". :D
beh in effetti ove occorre usare le stringhe a mano bassa il cpp e il namespace viene d'aiuto perchè li malloc te lo scordi
pero' se devi lavorare di fino o su controlli automatici il c è preferibile
vBulletin® v3.6.4, Copyright ©2000-2026, Jelsoft Enterprises Ltd.