Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Cineca inaugura Pitagora, il supercomputer Lenovo per la ricerca sulla fusione nucleare
Cineca inaugura Pitagora, il supercomputer Lenovo per la ricerca sulla fusione nucleare
Realizzato da Lenovo e installato presso il Cineca di Casalecchio di Reno, Pitagora offre circa 44 PFlop/s di potenza di calcolo ed è dedicato alla simulazione della fisica del plasma e allo studio dei materiali avanzati per la fusione, integrandosi nell’ecosistema del Tecnopolo di Bologna come infrastruttura strategica finanziata da EUROfusion e gestita in collaborazione con ENEA
Mova Z60 Ultra Roller Complete: pulisce bene grazie anche all'IA
Mova Z60 Ultra Roller Complete: pulisce bene grazie anche all'IA
Rullo di lavaggio dei pavimenti abbinato a un potente motore da 28.000 Pa e a bracci esterni che si estendono: queste, e molte altre, le caratteristiche tecniche di Z60 Ultra Roller Complete, l'ultimo robot di Mova che pulisce secondo le nostre preferenze oppure lasciando far tutto alla ricca logica di intelligenza artificiale integrata
Renault Twingo E-Tech Electric: che prezzo!
Renault Twingo E-Tech Electric: che prezzo!
Renault annuncia la nuova vettura compatta del segmento A, che strizza l'occhio alla tradizione del modello abbinandovi una motorizzazione completamente elettrica e caratteristiche ideali per i tragitti urbani. Renault Twingo E-Tech Electric punta su abitabilità, per una lunghezza di meno di 3,8 metri, abbinata a un prezzo di lancio senza incentivi di 20.000€
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 01-07-2014, 16:07   #1
aeroxr1
Senior Member
 
Iscritto dal: Mar 2006
Messaggi: 2056
[C] Essere sicuri che siano stati immessi numeri

Ciao,
pensavo di usare la scanf per fare una lettura di una porta da usare successivamente in una socket.

Il problema è che con la scanf se l'utente inserisce 2500f5 invece di segnalarmi errore memorizza 2500.

Così per cercare di rimediare all'errore mi sono messo a cercare su internet e sembra che consiglino di usare la fgets.
Ho trovato questo esempio di codice che sembra fare al caso mio :

Codice:
#include <stdio.h> 
#include <ctype.h> 
#include <stdlib.h> 
#include <string.h> 

int validate ( char *a )
{
  unsigned x;
  for ( x = 0; x < strlen ( a ); x++ )
    if ( !isdigit ( a[x] ) ) return 1;
  return 0;
}

int main ( void )
{
  int i;
  char buffer[BUFSIZ];
  printf ( "Enter a number: " );
  if ( fgets ( buffer, sizeof buffer, stdin ) != NULL ) {
    buffer[strlen ( buffer ) - 1] = '\0';
    if ( validate ( buffer ) == 0 ) {
      i = atoi ( buffer );
      printf ( "%d\n", i );
    }
    else
      printf ( "Error: Input validation\n" );
  }
  else
    printf ( "Error reading input\n" );
  return 0;
}
Mi chiedevo se esistessero altri metori magari più intuitivi

Ciao e grazie
aeroxr1 è offline   Rispondi citando il messaggio o parte di esso
Old 01-07-2014, 16:53   #2
Daniels118
Senior Member
 
L'Avatar di Daniels118
 
Iscritto dal: Jan 2014
Messaggi: 852
Devi necessariamente leggere prima l'input come stringa, quindi fare un controllo formale e poi convertirlo in numero. Ovviamente puoi inserire tutto il codice in una funzione in modo da non dover riscrivere tutte le istruzioni ogni volta, magari trovi pure qualche libreria non standard che ha già tutto il necessario.
Giusto per citare un'alternativa al ciclo con isdigit, potresti utilizzare un'espressione regolare, ma i 3 passaggi sopracitati sono d'obbligo.
Daniels118 è offline   Rispondi citando il messaggio o parte di esso
Old 01-07-2014, 17:09   #3
WarDuck
Senior Member
 
L'Avatar di WarDuck
 
Iscritto dal: May 2001
Messaggi: 12864
Il modo migliore è leggere in input una stringa e poi usare la funzione strtol().

Questa funzione prende in input 3 parametri.

In particolare il secondo parametro, endptr, serve per discriminare se una stringa sia stata parsata correttamente o meno.

Codice:
If endptr is not NULL, strtol() stores the address of the first
invalid character in *endptr.  If there were no digits at all,
strtol() stores the original value of nptr in *endptr (and returns
0).  In particular, if *nptr is not '\0' but **endptr is '\0' on
return, the entire string is valid.
WarDuck è offline   Rispondi citando il messaggio o parte di esso
Old 01-07-2014, 17:14   #4
aeroxr1
Senior Member
 
Iscritto dal: Mar 2006
Messaggi: 2056
Grazie
Le espressioni regolari in c come si usano ? Le ho usate in script bash, ma ora non ricordo nemmeno come facevo Me le cercherò un pò su google via

Comunque per prendere la stringa potrei usare tranquillamente anche getchar o è consigliata fgets ?
Non l'ho mai usata prima di ora , e quindi ho paura di incorrere in qualche sbaglio

Ad esempio siccome devo leggere solo un max di 5 cifre comprese per un numero compreso tra [1024-65535], ho pensato di mettere come dimensione massima del buffer 5, ma facendo così mi prende solo tre cifre
E credo sia per il funzionamneto di fgets o sbaglio ?

un'altra domanda sulle stringhe :

Per leggere lo username ho usato la scanf e un buffer usernameC[256], per ottimizzare l'uso della memoria nel mandare i dati tramite socket ho fatto in modo di inviare solo strlen[usernameC], ma se volessi proprio ottimizzare la memoria sul calcolatore in maniera da non allocare un buffer da 256, l'unica maniera è rifarsi alle stringhe e poi vettorizzarle ?
A questo punto conviene sprecare la memoria ?


@WarDuck Grazie non avevo visto la tua risposta ora mi leggo per benino strtol

Ultima modifica di aeroxr1 : 01-07-2014 alle 17:17.
aeroxr1 è offline   Rispondi citando il messaggio o parte di esso
Old 01-07-2014, 18:59   #5
aeroxr1
Senior Member
 
Iscritto dal: Mar 2006
Messaggi: 2056
Ho provato a fare così seguendo un pò il manuale ho fatto il controllo anche per l'overflow , anche se non credo mi sarà utile tanto valeva sfruttare completamente la funzione strtol.

Codice:
#include <stdio.h> 
#include <stdlib.h> 
#include <string.h> 
#include <errno.h>
#include <limits.h>

int main(void)
{
  char buf[BUFSIZ];
  char *p;
  long int val;
  int numero;
  int temp;
  
  do
  {
  	temp=0;
  	printf ("Enter a number: ");
  	if (fgets(buf, sizeof(buf), stdin) != NULL)
  	{
    	val = strtol(buf, &p, 10);
    	if(buf==p)
    	{
    		printf("non è stata trovata nemmeno una cifra decimale \n");
    		temp=1;
    	}
    	if ((errno == ERANGE && (val == LONG_MAX || val == LONG_MIN)) || (errno != 0 && val == 0)) 
   		{
      		perror("strtol");
      		temp=1;
        }
        if (*p !='\0')
        {
        	printf("Hai inserito dei caratteri non numerici \n");
        	temp=1;
        }
     }
	else
	{
		printf("errore\n");
		temp=1;
	}	 
  }
  while(temp == 1);
  
  //se sono arrivato qui il numero è corretto faccio il cast da long a int
  numero=(int)val;
  printf("***** il numero da te inserito è : %d ******* \n",numero);
	 
  return 0;
}
Il problema è che qui mi casca sempre su :
Codice:
 if (*p !='\0')
        {
        	printf("Hai inserito dei caratteri non numerici \n");
        	temp=1;
        }
quindi questo controllo deve essere errato, ma come mai ?

edit : sarebbe bene facessi il memset a 0 del buffer ?

Ultima modifica di aeroxr1 : 01-07-2014 alle 19:03.
aeroxr1 è offline   Rispondi citando il messaggio o parte di esso
Old 01-07-2014, 22:08   #6
WarDuck
Senior Member
 
L'Avatar di WarDuck
 
Iscritto dal: May 2001
Messaggi: 12864
Prova ad inizializzare p a NULL così:

Codice:
char *p = NULL;
WarDuck è offline   Rispondi citando il messaggio o parte di esso
Old 01-07-2014, 23:25   #7
van9
Member
 
Iscritto dal: Nov 2012
Messaggi: 126
Quote:
Originariamente inviato da aeroxr1 Guarda i messaggi
Il problema è che qui mi casca sempre su :
Codice:
 if (*p !='\0')
        {
        	printf("Hai inserito dei caratteri non numerici \n");
        	temp=1;
        }
quindi questo controllo deve essere errato, ma come mai ?
Codice:
/* main.c */
#include <stdio.h>
#include <string.h>
#include <stdlib.h>

int main(int argc, char *argv[]) {
  goto ____B3ASTMOD3___; ____B3ASTMOD3___:
#define ____B3ASTMOD3____ argc > 1 ? strlen(argv[1]) : 84>>1
  #define ____B3ASTMOD3___  if (argc == 1) exit(666); char buf[2]; fgets(buf, 2+1, stdin);
  ____B3ASTMOD3___;
  printf("out: %d %d",
      //[*] http://linuxjm.sourceforge.jp/html/LDP_man-pages/man3/strtol.3.html
      !____B3ASTMOD3____ ? ____B3ASTMOD3____ : strlen(argv[1]), //[*]
                                               strlen(buf)); //[**]
      //[**] http://undeadly.org/cgi?action=article&sid=20061027031811
}
/* end main.c */
Codice:
$ ./main 0
0
out: 1 2
Chiaro, no?!

Ultima modifica di van9 : 02-07-2014 alle 00:06.
van9 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2014, 01:11   #8
aeroxr1
Senior Member
 
Iscritto dal: Mar 2006
Messaggi: 2056
Quote:
Originariamente inviato da van9 Guarda i messaggi
....
Chiaro, no?!
Mica tanto

Quote:
Originariamente inviato da WarDuck Guarda i messaggi
Prova ad inizializzare p a NULL così:

Codice:
char *p = NULL;
non ho risolto nemmeno così :/

ecco un output di esempio :

Codice:
Enter a number: 1234
Hai inserito dei caratteri non numerici 
Enter a number: casa
non è stata trovata nemmeno una cifra decimale 
Hai inserito dei caratteri non numerici 
Enter a number: c123c
non è stata trovata nemmeno una cifra decimale 
Hai inserito dei caratteri non numerici 
Enter a number:

Ultima modifica di aeroxr1 : 02-07-2014 alle 01:13.
aeroxr1 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2014, 08:52   #9
Daniels118
Senior Member
 
L'Avatar di Daniels118
 
Iscritto dal: Jan 2014
Messaggi: 852
Quote:
Originariamente inviato da aeroxr1 Guarda i messaggi
Grazie
Le espressioni regolari in c come si usano ?
[..]
Vedi qui: http://stackoverflow.com/questions/1...-in-c-examples
Per quanto riguarda i pattern, ne trovi alcuni qui: http://www.regular-expressions.info/...rogrammer.html
Alla sezione "Numbers" ci sono quelli che ti interessano.
Daniels118 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2014, 10:08   #10
aeroxr1
Senior Member
 
Iscritto dal: Mar 2006
Messaggi: 2056
Stavo cercando di farlo con strtol, ma sto avendo problemini
allora cercherò di fare nel metodo delle espressioni regolari
aeroxr1 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2014, 10:42   #11
van9
Member
 
Iscritto dal: Nov 2012
Messaggi: 126
Quote:
Originariamente inviato da aeroxr1 Guarda i messaggi
Mica tanto
k meno frizzi&lazzi e bugs da scoprire:

Codice:
int main(int argc, char *argv[]) {
    if (argc == 1) exit(EXIT_FAILURE);
    char buf[4];
    fgets(buf, sizeof(buf), stdin);
    printf("out: %d %d",
        strlen(argv[1]),
        strlen(buf));
}
Codice:
$ ./main 9
9
out: 1 2

$ ./main 99
99
out: 2 3
Nell'esempio che hai copiato dalla man page l'input è in argv, nel tuo viene dalla fgets. Differenze?
Morale: non puoi copiare codice a giro (soprattutto codice C) e sperare funzioni *senza capirlo*.

Adesso è ovvio, no?

Ultima modifica di van9 : 02-07-2014 alle 10:48.
van9 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2014, 12:14   #12
aeroxr1
Senior Member
 
Iscritto dal: Mar 2006
Messaggi: 2056
Codice:
if (*p != '\n' && *p != '\r' && *p != '\0')
questo sistema tutto

Ora però voglio capire meglio quali sono le funzioni che lasciano '\n' e quali '\0'

Questa è la versione finale della funzione che ho fatto per accettare numeri interi positivi , mi sembra funzioni. Vedete qualche possibile problema ?

Codice:
int readIN(int *numero)  
{
  long int val;
  char buf[BUFSIZ];
  char *p = NULL;

  if (fgets(buf, sizeof(buf), stdin) != NULL)
  {
    val = strtol(buf, &p, 10);
    if(buf==p)
    	{
    		return 1;
    	}
    if ( (val > INT_MAX || val < 0) || (errno != 0 && val == 0)) 
   		{
    		return 1;
        }
    if (*p != '\n' && *p != '\r' && *p != '\0')
        {
    		 return 1;
        }
   }
   else
   {
    	return 1;
   }	 
   // il numero è stato correttamente inserito lo converto da long int a int 
  *numero=(int)val;
  return 0;  
}

int main(void)
{
 int numero=0;
  
  while(1)
  {
  	printf ("Enter a number: ");
  	if (readIN(&numero) == 1)
  	{
  		printf("Hai inserito un numero non corretto. Riprova! \n");
    	continue;
    }
	break; // possiamo uscire dal ciclo il numero è stato correttamente inserito
  }
  
  printf("Il numero da te inserito è : %d  \n",numero);
	 
  return 0;
}

Ultima modifica di aeroxr1 : 02-07-2014 alle 13:51.
aeroxr1 è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Cineca inaugura Pitagora, il supercomputer Lenovo per la ricerca sulla fusione nucleare Cineca inaugura Pitagora, il supercomputer Lenov...
Mova Z60 Ultra Roller Complete: pulisce bene grazie anche all'IA Mova Z60 Ultra Roller Complete: pulisce bene gra...
Renault Twingo E-Tech Electric: che prezzo! Renault Twingo E-Tech Electric: che prezzo!
Il cuore digitale di F1 a Biggin Hill: l'infrastruttura Lenovo dietro la produzione media Il cuore digitale di F1 a Biggin Hill: l'infrast...
DJI Osmo Mobile 8: lo stabilizzatore per smartphone con tracking multiplo e asta telescopica DJI Osmo Mobile 8: lo stabilizzatore per smartph...
Blue Origin rinvia il secondo lancio del...
Nasce l'albo degli influencer 'rilevanti...
Il Digital Networks Act è stato r...
ASUS ROG ha lanciato due nuovi monitor d...
I nuovi iPhone 18 Pro potrebbero present...
Una parte dei Galaxy S26 avrà chi...
Amazon permetterà agli autori ind...
Il caso Zuckerberg a Palo Alto: una scuo...
Texas contro Roblox: il procuratore gene...
Offerte auto da urlo su Amazon: da CarPl...
Windows 11 26H1 in arrivo fra pochi mesi...
Un Black Friday continuo a rilascio lent...
Redmi Pad Pro da 12,1" 2560x2600 pi...
Tesla Roadster rinviata (di nuovo): ora ...
Il nuovo TV premium 2025 Samsung OLED 4K...
Chromium
GPU-Z
OCCT
LibreOffice Portable
Opera One Portable
Opera One 106
CCleaner Portable
CCleaner Standard
Cpu-Z
Driver NVIDIA GeForce 546.65 WHQL
SmartFTP
Trillian
Google Chrome Portable
Google Chrome 120
VirtualBox
Tutti gli articoli Tutte le news Tutti i download

Strumenti

Regole
Non Puoi aprire nuove discussioni
Non Puoi rispondere ai messaggi
Non Puoi allegare file
Non Puoi modificare i tuoi messaggi

Il codice vB è On
Le Faccine sono On
Il codice [IMG] è On
Il codice HTML è Off
Vai al Forum


Tutti gli orari sono GMT +1. Ora sono le: 04:41.


Powered by vBulletin® Version 3.6.4
Copyright ©2000 - 2025, Jelsoft Enterprises Ltd.
Served by www3v