View Full Version : devo leggere una stringa con scanf....
allora...il problema è il seguente...
quando leggo una stringa con scanf("%s",stringa);
la scanf mi legge dallo stdin la sequenza di caratteri fino allo spazio e poi ci mette un \0 alla fine nelle stringa......ma se tipo voglio inserire una stringa con più parole separate da uno spazio? come devo fare? c'è qualcuno che ha già affrontato questo problema e magari l ha risolto? avevo pensato tipo a gets ma durante la compilazione mi dice che gets non deve essere utilizzata!?:) anche nel man di linux non son riuscito a trovare niente a parte qualcosa tipo l'utilizzo di [] ma non ho capito bene......
aiutoooooooo!:)
strano che legga fino allo spazio, io mi ricordavo di aver affrontato problemi simili a programmazione 1, ma mi sembrava che leggesse tutto fino al newline...
cmq non vedo perché gets non dovrebbe essere utilizzata: noi a prog 1 l'abbiamo usata e i nostri programmi sono stati compilati su Linux senza problemi :)
(inutile dire che io i miei li ho testati sempre su Windows, quindi dovrebbe andar bene su entrambi i sistemi operativi)
cosa dice esattamente il warning del compilatore quando usi gets? sei sicuro di aver incluso la libreria in cui è dichiarata?
DarkRevenge
15-06-2005, 23:15
La gets ha il problema di non controllare la lunghezza della stringa che viene effettivamente data in input, quindi viene sconsigliata in quanto può causare problemi di buffer overflow. :O
Per leggere una stringa è possibile utilizzare la funzione fgets (http://www.lilik.it/~mirko/gapil/gapilsu107.html) , alla quale deve essere passata anche la lunghezza massima della stringa (effettivamente ho riscontrato anche io gli stessi problemi con la scanf, ma sinceramente non so se c'è un modo per risolverli)...
prova ad usare la scnaf così:
scanf(" %[^\n]",str);
devi lasciare uno spazio davanti al %
La gets ha il problema di non controllare la lunghezza della stringa che viene effettivamente data in input, quindi viene sconsigliata in quanto può causare problemi di buffer overflow. :O
Per leggere una stringa è possibile utilizzare la funzione fgets (http://www.lilik.it/~mirko/gapil/gapilsu107.html) , alla quale deve essere passata anche la lunghezza massima della stringa (effettivamente ho riscontrato anche io gli stessi problemi con la scanf, ma sinceramente non so se c'è un modo per risolverli)... ma appunto, quel problema caratterizza anche scanf("%s", s); (non so se è quello che intendevi dire); e poi non vedo perché il compilatore dovrebbe rompere le OO :D se uno vuole usare gets usa gets :D se il programma è a scopi didattici va più che bene :p e poi sinceramente mi pare anche abbastanza difficile un problema di sicurezza dovuto a una gets: per mandare in crash il programma si suppone che un ipotetico cracker abbia già accesso locale allo stdin del programma, ma se è così ce ne sono a milioni di modi di far crashare il prog. :sofico:
RaouL_BennetH
16-06-2005, 01:19
ma appunto, quel problema caratterizza anche scanf("%s", s); (non so se è quello che intendevi dire); e poi non vedo perché il compilatore dovrebbe rompere le OO :D se uno vuole usare gets usa gets :D se il programma è a scopi didattici va più che bene :p e poi sinceramente mi pare anche abbastanza difficile un problema di sicurezza dovuto a una gets: per mandare in crash il programma si suppone che un ipotetico cracker abbia già accesso locale allo stdin del programma, ma se è così ce ne sono a milioni di modi di far crashare il prog. :sofico:
No, il problema con gets è un altro, ma partendo dal compilatore...
se compili sotto Linux con gcc, è il compilatore stesso che come warning ti avvisa che non dovresti usare la gets. Il problema di sicurezza della gets e del buffer overflow, è che non puoi in nessun modo controllare la lunghezza della stringa che prendi. Non a caso, l'uso ne è vivamente sconsigliato su moltissimi testi, e non a caso, gcc dice vivamente di non usarla :D
No, il problema con gets è un altro, ma partendo dal compilatore...
se compili sotto Linux con gcc, è il compilatore stesso che come warning ti avvisa che non dovresti usare la gets. Il problema di sicurezza della gets e del buffer overflow, è che non puoi in nessun modo controllare la lunghezza della stringa che prendi. Non a caso, l'uso ne è vivamente sconsigliato su moltissimi testi, e non a caso, gcc dice vivamente di non usarla :D ehm... l'avevo capito... -_-'
Ziosilvio
17-06-2005, 10:59
quando leggo una stringa con scanf("%s",stringa);
Cosa che qualunque manuale recente sconsiglia di fare.
la scanf mi legge dallo stdin la sequenza di caratteri fino allo spazio e poi ci mette un \0 alla fine nelle stringa
Ma senza preoccuparsi che lo spazio allocato sia sufficiente.
ma se tipo voglio inserire una stringa con più parole separate da uno spazio?
Usa fgets.
Questo per leggere qualunque stringa.
scanf("%[A-Za-Z0-9 ]", %s);
Con questo prendi tutti i caratteri che cono lettere maiuscole, minuscole, numeri e spazi :)
Se ti servono altri caratteri, come la punteggiatura, aggiungili all'interno delle parentesi quadre :9
ho fatto questa routine e sembra funzionare...
int leggi_stringa(char *stringa,int n){
char ch; /*caratte di lettura */
int i=0; /*contatore di stringa*/
if (!stringa) return(-1);/*nel caso in cui la stringa non era stata allocata*/
if (fflush(stdin)==EOF) printf("c'e' stato un errore nella pulizia dello stdin\n");
/*questa funzione a volte basta a pulire lo stdin altre volte lo stdin si pulisce solo con il metodo adottato successivamente*/
/*leggo da stdin fino a che non leggo un newline, newline precedenti verranno ignorati grazie all'utilizzo del controllo i=0*/
while(((ch=getchar())!='\n')&&(i<n)||(i==0)){
if (ch!='\n'){stringa[i]=ch; i++;}/*se non sono una fine stringa memorizza in stringa[i]*/
}
if (i==n){ /*controllo l'overflow, in tal caso ho troncato lo stdin*/
while(getchar()!='\n');/*svuoto lo stdin di caratteri immondizia*/
stringa[n-1]='\0';
return(-2);
}
if (i+1==n) i--;
stringa[i+1]='\0';
return(0);
}
praticamente con getchar prendo tutti i caratteri da stdin fino al newline.
nel caso in cui ci siano newline vecchi nello stdin e cose vecchie pulisco lo stdin
che ne dite?
prova ad usare fflush(stdin)
RaouL_BennetH
19-06-2005, 13:06
ho fatto questa routine e sembra funzionare...
praticamente con getchar prendo tutti i caratteri da stdin fino al newline.
nel caso in cui ci siano newline vecchi nello stdin e cose vecchie pulisco lo stdin
che ne dite?
che ripulire il buffer credo sia cosa buona e giusta :)
Usa la fgets che è piu sicura (mi è stata consigliata qua sul forum)
fgets(stringa,MAXBUFF,stdin);
Ziosilvio
19-06-2005, 19:54
praticamente con getchar prendo tutti i caratteri da stdin fino al newline.
nel caso in cui ci siano newline vecchi nello stdin e cose vecchie pulisco lo stdin
che ne dite?
Che lo standard non definisce il comportamento di fflush su uno stream di input.
E che ch deve essere un int, e non un char (getchar può restituire EOF, che è un valore int diverso da qualunque char).
vBulletin® v3.6.4, Copyright ©2000-2026, Jelsoft Enterprises Ltd.