Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Panasonic 55Z95BEG cala gli assi: pannello Tandem e audio senza compromessi
Panasonic 55Z95BEG cala gli assi: pannello Tandem e audio senza compromessi
Con un prezzo di 2.999 euro, il Panasonic Z95BEG entra nella fascia ultra-premium dei TV OLED: pannello Primary RGB Tandem, sistema di raffreddamento ThermalFlow, audio Technics integrato e funzioni gaming avanzate lo pongono come un punto di riferimento
HONOR Magic V5: il pieghevole ultra sottile e completo! La recensione
HONOR Magic V5: il pieghevole ultra sottile e completo! La recensione
Abbiamo provato per diverse settimane il nuovo Magic V5 di HONOR, uno smartphone pieghevole che ci ha davvero stupito. Il device è il più sottile (solo 4.1mm) ma non gli manca praticamente nulla. Potenza garantita dallo Snapdragon 8 Elite, fotocamere di ottima qualità e batteria in silicio-carbonio che garantisce un'ottima autonomia. E il Prezzo? Vi diciamo tutto nella nostra recensione completa.
Recensione Google Pixel 10 Pro XL: uno zoom 100x assurdo sempre in tasca (e molto altro)
Recensione Google Pixel 10 Pro XL: uno zoom 100x assurdo sempre in tasca (e molto altro)
Google Pixel 10 Pro XL è il top di gamma della serie Pixel, presentando un ampio display Super Actua da 6.8 pollici insieme alle novità della serie, fra cui la ricarica wireless magnetica Pixelsnap e le nuove funzionalità AI avanzate. Il comparto fotografico include un sistema a tripla fotocamera con zoom Pro Res fino a 100x, mentre il processore Tensor G5 con 16GB di RAM garantisce prestazioni percepite molto elevate su Android.
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 02-07-2011, 11:15   #1
informatico91
Member
 
Iscritto dal: Sep 2009
Messaggi: 41
[C] invio segnalazione errori all'uscita del main

Ciao a tutti stavo svolgendo un programma c con la seguente traccia:

Gioco dell’impiccato
Si realizzi un programma in linguaggio C che permetta di giocare al gioco dell’impiccato. Il
gioco si svolge tra due giocatori: il primo giocatore inserisce la parola segreta da indovinare,
mentre il secondo la deve indovinare.
Il secondo giocatore conosce la lunghezza della parola segreta, e ad ogni tentativo specifica
una lettera (si considerino le lettere maiuscole equivalenti alle minuscole): se tale
lettera compare nella parola, il programma indica in quali posizioni, altrimenti il tentativo
è considerato un errore. Il gioco termina quando il secondo giocatore ha indovinato tutte
le lettere della parola (ed in tal caso egli vince) oppure quando ha totalizzato 10 errori (nel
qual caso perde).



Codice:
Codice:
#include<stdio.h>

int main()
{
      char parola_g1[] = {0};
      int i, j, k, lunghezza_p1, uscita = 0, fine = 0, tent = 0;
      char car1, car2;
      
      printf("Giocatore 1: Inserisci la parola segreta: ");
      scanf("%s", parola_g1); //in questo caso poichè nel vettore viene memorizzata una parola e non una stringa si può usare lo
                           //scanf anzichè la funzione gets. Anche perchè scanf accoda alla fine della stringa anche il carattere nullo
      
      for ( i = 0; parola_g1[i] != '\0'; i++ )
      {
          lunghezza_p1++;
      }
      
      int posizioni[lunghezza_p1];
      char parola_g2[lunghezza_p1];
     
      for ( i = 0; i < lunghezza_p1; i++) 
      {
          parola_g2[i] = '-';
      }
      
      printf("\nGiocatore 2: La parola che devi indovinare e' formata da %d caratteri.\nInserisci la tua parola. Hai 10 tentativi!!!\n ", lunghezza_p1);
      
      for ( i = 0; i < 10; i++ ) // for per i tentativi massimi
      {
          uscita = 0;
          fine = 0;
          
          
          for ( j = 0; j < lunghezza_p1; j++) //Azzera gli elementi della matrice delle posizioni. Questo va fatto in modo tale che
          {                                   // ogni volta che rientra nel ciclo più esterno si sappia la posizione delle lettere trovate
              posizioni[j] = 0;
          }
          fflush(stdin);
          printf("\nTentativo %d. Inserisci una lettera: ", i + 1);
          scanf("%c", &car1);
          
          /*Considera i casi maiuscolo e minuscolo*/
          if ( car1 >= 97 ) 
          {
               car2 = car1 - 32;
          }
          else if ( car1 <= 90 )
          {
               car2 = car1 + 32;
          }
          
          
      /*Scorre la parola alla ricerca di caratteri uguali a quello immesso*/          
          for ( j = 0; j < lunghezza_p1; j++ )
          {
              if ( parola_g1[j] == car1 || parola_g1[j] == car2)
              {
                   posizioni[j]++;
                   parola_g2[j] = car1;
              }
              else if ( parola_g1[j] != car1 || parola_g1[j] != car2 )
              {
                   uscita++;
              }
              //printf("%d\n", uscita);

          }
          
          
          if ( uscita == lunghezza_p1 )
              {
                  printf("La lettera digita non è stata trovata\n");
              }
          else
          {
              i--;// Il tentativo corrente sarà ancora valido
              
              printf("La lettera '%c' e' stata trovata nella/e posizione/i: ", car1);
              for ( j = 0; j < lunghezza_p1; j++ )
              {
                  if ( posizioni[j] != 0 )
                  {
                     printf("%d, ", j + 1);
                  }
              }
              printf("\n");
          
              for ( j = 0; j < lunghezza_p1; j++ )
              {
                  printf("%c", parola_g2[j]);
              }
              
              for ( j = 0; j < lunghezza_p1; j++) 
              {
                if ( parola_g2[j] != '-' )
                {
                   fine++;
                }    
              }
              //printf("%d\n", uscita);
              if ( fine == lunghezza_p1 )
              {
                 printf("\nComplimenti hai indovinato la parola!!!!");
                 i = lunghezza_p1 + 1;
                 break; // questo serve per evitare che venga svolta la successiva istruzione if e non è questa istruzione a dare
                        // il problema dell'invio segnalazione errori
              }
          }
          
          
          if ( i == 9 )
          {
             printf("Hai superato il numero massimo di tentativi per indovinare la parola. Rigioca sarai piu' fortunato!!");
           }
      }
      
      printf("\nProgramma terminato");
      return 0;
}
Ho fatto diversi tentativi di esecuzione e l'algoritmo sembra funzionare correttamente. Succede però che qualsiasi strada prendo (o decido di indovinare la parola o di finire i tentavi) ogni volta che termina il main mi esce la solita finestrella di windows invia segnalazione errori.
Secondo voi perchè si comporta così???
informatico91 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2011, 13:13   #2
AngeL)
Senior Member
 
L'Avatar di AngeL)
 
Iscritto dal: May 2006
Città: Salerno
Messaggi: 936
Quote:
Originariamente inviato da informatico91 Guarda i messaggi
Codice:
#include<stdio.h>

int main()
{
      char parola_g1[] = {0}; //errore: non allochi spazio per l'array! riservati almeno 20 caratteri dichiarando "char parola_g1[20];"
      int i, j, k, lunghezza_p1 = 0, uscita = 0, fine = 0, tent = 0;
      char car1, car2;
      
      printf("Giocatore 1: Inserisci la parola segreta: ");
      scanf("%s", parola_g1); //qui avevi riservato soltanto un byte di spazio, non sai dove vai a scrivere in memoria!
      
      for ( i = 0; parola_g1[i] != '\0'; i++ )
      {
          lunghezza_p1++; //errore: stai incrementando lunghezza_p1 ma non l'hai inizializzato a 0!
      }
      
      int posizioni[lunghezza_p1];
      char parola_g2[lunghezza_p1];
     
      for ( i = 0; i < lunghezza_p1; i++) 
      {
          parola_g2[i] = '-';
      }
      
      printf("\nGiocatore 2: La parola che devi indovinare e' formata da %d caratteri.\nInserisci la tua parola. Hai 10 tentativi!!!\n ", lunghezza_p1);
      
      for ( i = 0; i < 10; i++ ) // for per i tentativi massimi
      {
          uscita = 0;
          fine = 0;
          
          
          for ( j = 0; j < lunghezza_p1; j++) //Azzera gli elementi della matrice delle posizioni. Questo va fatto in modo tale che
          {                                   // ogni volta che rientra nel ciclo più esterno si sappia la posizione delle lettere trovate
              posizioni[j] = 0;
          }
          fflush(stdin);
          printf("\nTentativo %d. Inserisci una lettera: ", i + 1);
          scanf("%c", &car1);
          
          /*Considera i casi maiuscolo e minuscolo*/
          if ( car1 >= 97 ) 
          {
               car2 = car1 - 32;
          }
          else if ( car1 <= 90 )
          {
               car2 = car1 + 32;
          }
          
          
      /*Scorre la parola alla ricerca di caratteri uguali a quello immesso*/          
          for ( j = 0; j < lunghezza_p1; j++ )
          {
              if ( parola_g1[j] == car1 || parola_g1[j] == car2)
              {
                   posizioni[j]++;
                   parola_g2[j] = car1;
              }
              else if ( parola_g1[j] != car1 || parola_g1[j] != car2 )
              {
                   uscita++;
              }
              //printf("%d\n", uscita); //il sistema che usi è abbastanza contorto, ma cerchiamo di correggere prima gli errori, poi di ottimizzare

          }
          
          
          if ( uscita == lunghezza_p1 )
              {
                  printf("La lettera digita non è stata trovata\n");
              }
          else
          {
              i--;// Il tentativo corrente sarà ancora valido
              
              printf("La lettera '%c' e' stata trovata nella/e posizione/i: ", car1);
              for ( j = 0; j < lunghezza_p1; j++ )
              {
                  if ( posizioni[j] != 0 )
                  {
                     printf("%d, ", j + 1);
                  }
              }
              printf("\n");
          
              for ( j = 0; j < lunghezza_p1; j++ ) //qui stampi una stringa carattere per carattere; non ti conviene aggiungerci il terminatore '\0' e stamparla direttamente come stringa?
              {
                  printf("%c", parola_g2[j]);
              }
              
              for ( j = 0; j < lunghezza_p1; j++) 
              {
                if ( parola_g2[j] != '-' )
                {
                   fine++;
                }    
              }
              //printf("%d\n", uscita);
              if ( fine == lunghezza_p1 )
              {
                 printf("\nComplimenti hai indovinato la parola!!!!");
                 i = lunghezza_p1 + 1; //perchè? basta il break per uscire, e poi se la parola è composta da meno di 10 caratteri, con questa riga di codice fornisci tentativi infiniti al giocatore, no? Forse volevi fare i = (tentativi) + 1?
                 break; // questo serve per evitare che venga svolta la successiva istruzione if e non è questa istruzione a dare
                        // il problema dell'invio segnalazione errori
              }
          }
          
          
          if ( i == 9 )
          {
             printf("Hai superato il numero massimo di tentativi per indovinare la parola. Rigioca sarai piu' fortunato!!");
           }
      }
      
      printf("\nProgramma terminato");
      return 0;
}
Ho fatto diversi tentativi di esecuzione e l'algoritmo sembra funzionare correttamente. *tossisce* Succede però che qualsiasi strada prendo (o decido di indovinare la parola o di finire i tentavi) ogni volta che termina il main mi esce la solita finestrella di windows invia segnalazione errori.
Secondo voi perchè si comporta così???
Probabilmente per la gestione della memoria non proprio ottimale, vai a scrivere in aree non allocate, incrementi valori non inizializzati... prova ad aggiustare queste cosine, poi c'è anche l'algoritmo da rivedere.

Ultima modifica di AngeL) : 02-07-2011 alle 13:17.
AngeL) è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2011, 13:21   #3
informatico91
Member
 
Iscritto dal: Sep 2009
Messaggi: 41
Quote:
Originariamente inviato da AngeL) Guarda i messaggi
Probabilmente per la gestione della memoria non proprio ottimale, vai a scrivere in aree non allocate, incrementi valori non inizializzati... prova ad aggiustare queste cosine, poi c'è anche l'algoritmo da rivedere.
Sospettavo che il problema derivasse dalla memoria..infatti quell'errore mi esce spesso quando dimentico di mettere l'uppersand nello scanf..comunque so di non essere molto esperto e di buttare giù i programmi con gli algoritmi che mi vengono al momento..
Cosa intendi dicendo "vai a scrivere in aree non allocate, incrementi valori non inizializzati"?? mi faresti vedere dove succedono ste cose nel codice??...
informatico91 è offline   Rispondi citando il messaggio o parte di esso
Old 02-07-2011, 15:13   #4
AngeL)
Senior Member
 
L'Avatar di AngeL)
 
Iscritto dal: May 2006
Città: Salerno
Messaggi: 936
Quote:
Originariamente inviato da informatico91 Guarda i messaggi
Cosa intendi dicendo "vai a scrivere in aree non allocate, incrementi valori non inizializzati"?? mi faresti vedere dove succedono ste cose nel codice??...
Quando dichiari
Codice:
char parola_g1[] = {0};
il compilatore ti riserva UN byte di spazio, e lo setta a 0. Quando poi vai a scriverci, finisci nelle aree di memoria successive (il c++ non lo vieta) e potresti sovrascrivere qualche altra variabile, o comunque fare qualche casotto

Quando invece dichiari
Codice:
int lunghezza_p1;
stai dichiarando una variabile, ma non ne setti il valore. Il compilatore ti riserva della memoria, ma non la tocca: se contiene zero la tua variabile varrà zero; se contiene qualcosa, la tua variabile sarà quel qualcosa! E quando fai
Codice:
lunghezza_p1++;
aumenti di 1 quella quantità, ma nulla ti garantisce che sia 0!
AngeL) è offline   Rispondi citando il messaggio o parte di esso
Old 03-07-2011, 10:49   #5
informatico91
Member
 
Iscritto dal: Sep 2009
Messaggi: 41
ciao ho seguito il tuo consiglio e riscritto parte del programma..ora funziona perfettamente e non compare più la finestra di invio segnalazione errori...pazienza per l'algoritmo magari non proprio ottimale ma per il momento mi basta che escano i programmi


Codice:
/*Gioco dell’impiccato
Si realizzi un programma in linguaggio C che permetta di giocare al gioco dell’impiccato. Il
gioco si svolge tra due giocatori: il primo giocatore inserisce la parola segreta da indovinare,
mentre il secondo la deve indovinare.
Il secondo giocatore conosce la lunghezza della parola segreta, e ad ogni tentativo specifica
una lettera (si considerino le lettere maiuscole equivalenti alle minuscole): se tale
lettera compare nella parola, il programma indica in quali posizioni, altrimenti il tentativo
è considerato un errore. Il gioco termina quando il secondo giocatore ha indovinato tutte
le lettere della parola (ed in tal caso egli vince) oppure quando ha totalizzato 10 errori (nel
qual caso perde).
*/
#include<stdio.h>

int main()
{
      char parola_g1[20] = {0};
      char parola_g2[20] = {0};
      int posizioni[20] = {0};
      int i, j, k, lunghezza = 0, uscita = 0, fine = 0;
      char car1, car2;
      
      printf("Giocatore 1: Inserisci la parola segreta: ");
      scanf("%s", parola_g1); //in questo caso poichè nel vettore viene memorizzata una parola e non una stringa si può usare lo
                           //scanf anzichè la funzione gets. Anche perchè scanf accoda alla fine della stringa anche il carattere nullo
      
      for ( i = 0; parola_g1[i] != '\0'; i++ )
      {
          lunghezza++;
      }
 
      for ( i = 0; i < lunghezza; i++) 
      {
          parola_g2[i] = '-';
      }
      
      printf("\nGiocatore 2: La parola che devi indovinare e' formata da %d caratteri.\nInserisci la tua parola. Hai 10 tentativi!!!\n ", lunghezza);
      
      for ( i = 0; i < 10; i++ ) // for per i tentativi massimi
      {
          uscita = 0;
          fine = 0;
          
          
          for ( j = 0; j < lunghezza; j++) //Azzera gli elementi della matrice delle posizioni. Questo va fatto in modo tale che
          {                                   // ogni volta che rientra nel ciclo più esterno si sappia la posizione delle lettere trovate
              posizioni[j] = 0;
          }
          fflush(stdin);
          printf("\nTentativo %d. Inserisci una lettera: ", i + 1);
          scanf("%c", &car1);
          
          /*Considera i casi maiuscolo e minuscolo*/
          if ( car1 >= 97 ) 
          {
               car2 = car1 - 32;
          }
          else if ( car1 <= 90 )
          {
               car2 = car1 + 32;
          }
          
          
      /*Scorre la parola alla ricerca di caratteri uguali a quello immesso*/          
          for ( j = 0; parola_g1[j] != '\0'; j++ )
          {
              if ( parola_g1[j] == car1 || parola_g1[j] == car2)
              {
                   posizioni[j]++;
                   parola_g2[j] = car1;
              }
              else if ( parola_g1[j] != car1 || parola_g1[j] != car2 )
              {
                   uscita++;
              }
              //printf("%d\n", uscita);

          }
          
          
          if ( uscita == lunghezza )
              {
                  printf("La lettera digita non è stata trovata\n");
              }
          else
          {
              i--;// Il tentativo corrente sarà ancora valido
              
              printf("La lettera '%c' e' stata trovata nella/e posizione/i: ", car1);
              for ( j = 0; j < lunghezza; j++ )
              {
                  if ( posizioni[j] != 0 )
                  {
                     printf("%d, ", j + 1);
                  }
              }
              printf("\n");
          
              for ( j = 0; j < lunghezza; j++ )
              {
                  printf("%c", parola_g2[j]);
              }
              
              for ( j = 0; j < lunghezza; j++) 
              {
                if ( parola_g2[j] != '-' )
                {
                   fine++;
                }    
              }
              //printf("%d\n", uscita);
              if ( fine == lunghezza )
              {
                 printf("\nComplimenti hai indovinato la parola!!!!");
                 i = lunghezza + 1;
                 break; // questo serve per evitare che venga svolta la successiva istruzione if e non è questa istruzione a dare
                        // il problema dell'invio segnalazione errori
              }
          }
          
          
          if ( i == 9 )
          {
             printf("Hai superato il numero massimo di tentativi per indovinare la parola. Rigioca sarai piu' fortunato!!");
           }
      }
      
      printf("\nProgramma terminato");
      return 0;
}
Se non ricordo male l'unica cosa che ho fatto è quella di mettere i due vettori di caratteri a dimensione fissa e ho semplicemente adattato il resto del programma a questa modifica. Grazie tante per i tuoi suggerimenti
informatico91 è offline   Rispondi citando il messaggio o parte di esso
Old 03-07-2011, 11:48   #6
AngeL)
Senior Member
 
L'Avatar di AngeL)
 
Iscritto dal: May 2006
Città: Salerno
Messaggi: 936
Quote:
Originariamente inviato da informatico91 Guarda i messaggi
Se non ricordo male l'unica cosa che ho fatto è quella di mettere i due vettori di caratteri a dimensione fissa e ho semplicemente adattato il resto del programma a questa modifica. Grazie tante per i tuoi suggerimenti
Felicissimo d'essere stato d'aiuto

Qualche altra piccola ottimizzazione che mi viene in mente:

- Anzichè utilizzare tutto quello stratagemma della variabile lunghezza, basta chiamare la funzione
Codice:
strlen(parola_g1); //dichiarata in <string.h>
per ottenerne la lunghezza.
- Le variabili i e j possono anche essere dichiarate nel corpo del ciclo che le usa, non ha senso dichiararle all'inizio del main; k, poi, non la usi proprio! :P
- Il sistema che hai implementato con le variabili uscita e fine è un po' astruso: prova a percorrere il programma passo passo, troverai certamente dei modi per ottimizzare il ragionamento e quindi l'algoritmo
- Il metodo che usi per distinguere tra maiuscole e minuscole non è errato, ma è dispendioso in termini di tempo e memoria. Potresti convertire direttamente sia la stringa sia il carattere in minuscolo (o maiuscolo) con le funzioni tolower(carattere) o toupper(carattere) dell'header <ctype.h> e poi considerare solo quel caso.
- Al posto del for per riempire parola_g2 di trattini, prova
Codice:
memset(parola_g2, '-', strlen(parola_g1));

- Al posto di usare un for come ciclo principale del programma, con tutti i problemi che ne conseguono per quanto riguarda i tentativi, io userei un while(tentativi < 10) ed incrementerei i tentativi in caso di errore.

Come avrai notato sono un po' pignolo, il codice andava già benissimo prima; questi prendili come "consigli" per una futura ottimizzazione!
AngeL) è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Panasonic 55Z95BEG cala gli assi: pannello Tandem e audio senza compromessi Panasonic 55Z95BEG cala gli assi: pannello Tande...
HONOR Magic V5: il pieghevole ultra sottile e completo! La recensione HONOR Magic V5: il pieghevole ultra sottile e co...
Recensione Google Pixel 10 Pro XL: uno zoom 100x assurdo sempre in tasca (e molto altro) Recensione Google Pixel 10 Pro XL: uno zoom 100x...
Lenovo IdeaPad Slim 3: un notebook Snapdragon X economico Lenovo IdeaPad Slim 3: un notebook Snapdragon X ...
Recensione OnePlus Watch 3 43mm: lo smartwatch che mancava per i polsi più piccoli Recensione OnePlus Watch 3 43mm: lo smartwatch c...
2 super portatili OLED in sconto: risolu...
Samsung Galaxy S25 Edge e Ultra in super...
La nuova Xbox sarà più pot...
In Svizzera la posta e la spesa te la co...
OnePlus 15: la ricarica diventerà ancora...
Roborock QV 35A oggi al minimo storico: ...
Smantellata Streameast: la più gr...
Il nuovo tri-fold di Samsung è vi...
Samsung tornerà alla strategia de...
Dalla Svizzera arriva un nuovo LLM open ...
Marshall annuncia novità per la g...
Occhio alle batterie dei prossimi smartp...
'Così gli sviluppatori smetterann...
Il futuro dei chip sarà (molto) p...
I ladri colpiscono le guardie: dei cyber...
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: 08:19.


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