Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Plaud NotePin S, il registratore IA si fa indossabile (ma è facile da perdere)
Plaud NotePin S, il registratore IA si fa indossabile (ma è facile da perdere)
Quattro modi di indossarlo, stessa app del Plaud Note Pro e integrazione con il desktop. Il registratore IA da indossare di Plaud eccelle in mobilità, ma resta vincolato all'abbonamento ed è facile da perdere
Redmi Watch 6 in prova: lo smartwatch con ampio display da 2000 nit a meno di 100 euro
Redmi Watch 6 in prova: lo smartwatch con ampio display da 2000 nit a meno di 100 euro
Xiaomi ha portato Redmi Watch 6 anche sul mercato italiano, puntando su un display AMOLED da 2,07 pollici con picco di luminosità a 2000 nit, frame in alluminio da 9,9mm e un'autonomia dichiarata di 12 giorni. Lo smartwatch gira su HyperOS 3 e integra GPS, Bluetooth 5.4 e oltre 150 sport mode. Il tutto a meno di 100 euro
Mad Catz M.M.O. 7+: lo stesso DNA del R.A.T. 8+ ADV, ma con molti più pulsanti
Mad Catz M.M.O. 7+: lo stesso DNA del R.A.T. 8+ ADV, ma con molti più pulsanti
Con 22 tasti, il pulsante 5D, lo Shift Mode e il sensore PixArt 3395 da 26.000 DPI, il nuovo mouse wireless di Mad Catz si rivolge in modo preciso ai giocatori di MMO e RPG. Ma chi conosce già il R.A.T. 8+ ADV si accorgerà subito di quanto i due prodotti condividano, e di dove invece divergono
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 22-03-2007, 21:24   #1
Capua
Junior Member
 
Iscritto dal: Dec 2006
Messaggi: 15
[C] Chiarimento liste concatenate

Qualcuno può commentarmi questo esercizio riga per riga in cui creo una lista concatenata in cui gli elementi sono dati in input fino all'inserimento di 9999? Vorrei cercare di capire meglio l'esercizio visto che non è completamente opera mia...Vi ringrazio anticipatamente..

/* STRUTTURA GLOBALE */

typedef struct lista {
int val;
struct lista *next;
} LISTA;


/* FUNZIONE MAIN */

int main (void){

LISTA *testa,*nuovo,*coda;

testa=NULL;
int elemento;

while (elemento!=9999){
printf("inserisci elemento, (9999 per terminare) " );
scanf("%d",&elemento);

if (elemento!=9999){

/* CREAZIONE LISTA */

nuovo=(LISTA*)malloc (sizeof(LISTA));
nuovo->val=elemento;
if(testa==NULL){

testa=nuovo;
nuovo->next=NULL;
coda=nuovo;
}

else {
nuovo->next=NULL;
coda->next=nuovo;
coda=nuovo;

}
}
}

printf("gli elementi dati in input sono: ");
stampa_lista(testa);
printf("\n");

return 0;
}
Capua è offline   Rispondi citando il messaggio o parte di esso
Old 23-03-2007, 01:08   #2
sottovento
Senior Member
 
L'Avatar di sottovento
 
Iscritto dal: Nov 2005
Città: Texas
Messaggi: 1722
Codice:
typedef struct lista {
int val;
struct lista *next;
} LISTA;
Qui dichiari il tipo LISTA, il quale e' adatto a contenere un valore intero e ad avere un puntatore ad un eventuale elemento successivo, il quale e' dello stesso tipo.
Come avrai ben capito, questa struttura si utilizza quando non sai a priori il numero di elementi che puoi avere. Ovviamente ha vantaggi e svantaggi, non e' questa la sede.
In generale, per utilizzare una struttura del genere, avrai bisogno di memorizzare il puntatore al primo elemento della lista. Da questo, poi, potrai facilmente scorrere tutta la lista seguendo i puntatori next.

Codice:
LISTA *testa,*nuovo,*coda;
Eccoli qui! I nomi non sono stati scelti a caso. Anzi, sono ben scelti. Stando ai nomi, ti aspetterai che testa memorizzi sempre l'indirizzo del primo elemento della lista (i.e. quello da cui potrai sempre scandire tutta la lista seguendo i next).
Potrai gia' immaginare che nuovo sara' l'elemento nuovo che vuoi inserire nella lista, e coda sara' il puntatore all'ultimo elemento.
Perche' ti serve il puntatore all'ultimo elemento, visto che con il puntatore "testa" posso scandire tutta la lista? Evidentemente perche' il programma vorra' inserire i nuovi elementi in coda, e scandire tutte le volte la lista potrebbe essere piuttosto oneroso (devi sempre far passare tutti gli elementi....). Meglio ricordarsi qual e' l'ultimo elemento, no?

Codice:
testa=NULL;
All'inizio la lista e' vuota, ed il puntatore alla testa e' ovviamente NULL.

Codice:
while (elemento!=9999){
printf("inserisci elemento, (9999 per terminare) " ); 
scanf("%d",&elemento);
Qui possiamo vedere il primo errore. Il programmatore aveva intenzione di fare un ciclo che fosse eseguito fintanto che elemento non assume il valore 9999. Prima di entrare in questo ciclo e' percio' necessario dare un valore ad elemento che sia diverso da 9999. Il programmatore se l'e' dimenticato. Siccome se non dai alcun valore il contenuto e' indeterminato (potrebbe esserci dentro qualsiasi cosa) c'e' buona probabilita' che non sia 9999, pertanto il programma SEMBRA funzionare correttamente.
Non funzionera' correttamente quando dovrai farlo vedere al cliente finale

Codice:
if (elemento!=9999){
Certo, se elemento == 9999 non fai niente. Alla prossima iterazione, uscirai

Codice:
nuovo=(LISTA*)malloc (sizeof(LISTA)); 
nuovo->val=elemento;
Qui possiamo vedere il secondo, subdolo, errore.
La malloc() ti alloca memoria dinamica. Signfica che, quando questa operazione si conclude con successo, la variabile nuovo assumera' un nuovo valore: puntera' ad un'area di memoria valida e sufficientemente grande per contenere il numero di bytes che hai specificato nel parametro.
Per esempio,
Codice:
char *p;
p = (char *)malloc(3);
hai la certezza che, se l'operazione si conclude con successo, p conterra' l'indirizzo di un'area di memoria che puo' contenere almeno 3 byte.
La scrittura (LISTA *) oppure (char *) si chiama type cast ed e' una semplice conversione di tipo. Siccome la malloc() non puo' sapere come utilizzerai quella memoria, ti riporta un puntatore "neutro" (la maggior parte dei nuovi sistemi riporta un puntatore a void, quelli vecchi a char).
Con questo type cast semplicemente dici: "utilizzo questa memoria per puntare ad un tipo LISTA", che poi e' il tipo della variabile a cui assegni.

Si parlava di errore: infatti ho sempre detto "se l'operazione si conclude con successo". Nel caso non si concluda con successo, la malloc() ti riporta NULL, pertanto l'istruzione
Codice:
nuovo->val=elemento;
mandera' in crash l'applicazione. Questo e' un "red-eyes crash", nel senso che spesso, nei programmi piuttosto grandi, ti obbliga a far la nottata a cercarlo.
Metti sempre il controllo: se non si riesce ad allocare memoria, piuttosto che niente, stampa un messaggio di errore ed esci.

Codice:
if(testa==NULL){

testa=nuovo; 
nuovo->next=NULL;
coda=nuovo;
}
OK! E' la prima volta che esegui il ciclo. La lista e' ancora vuota, pertanto l'elemento che hai inserito e' la testa della lista. testa (puntatore al primo elemento) pertanto ha lo stesso indirizzo del nuovo elemento (i.e. testa=nuovo).
D'altronde anche coda ha lo stesso indirizzo del nuovo elemento, in quanto la lista e' composta da un solo elemento (i.e. coda=nuovo).
L'elemento successivo? Non esiste, hai inserito un solo elemento, pertanto nuovo->next e' messo a NULL

Codice:
else {
nuovo->next=NULL;
coda->next=nuovo;
coda=nuovo; 

}
Abbiamo detto che si vuole inserire in coda. Questa e' la conferma. Siccome abbiamo visto che l'if stabilisce se stai inserendo il primo elemento, l'else significa che la tua lista ha almeno gia' un elemento. Esiste almeno la testa.
Quindi, siccome vogliamo inserire in coda, possiamo mettere:
- nuovo->next a NULL (e' l'ultimo, dietro di lui il vuoto);
- coda->next (cioe' quello che prima era l'ultimo elemento) ora punta a questo qui, appena inserito.
- coda = nuovo (questo e' il nuovo ultimo dei fratelli).

Scusa se sono stato prolisso
__________________
In God we trust; all others bring data
sottovento è offline   Rispondi citando il messaggio o parte di esso
Old 23-03-2007, 19:40   #3
Capua
Junior Member
 
Iscritto dal: Dec 2006
Messaggi: 15
Grazie mille, me l'hai commentato alla perfezione!
Capua è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Plaud NotePin S, il registratore IA si fa indossabile (ma è facile da perdere) Plaud NotePin S, il registratore IA si fa indoss...
Redmi Watch 6 in prova: lo smartwatch con ampio display da 2000 nit a meno di 100 euro Redmi Watch 6 in prova: lo smartwatch con ampio ...
Mad Catz M.M.O. 7+: lo stesso DNA del R.A.T. 8+ ADV, ma con molti più pulsanti Mad Catz M.M.O. 7+: lo stesso DNA del R.A.T. 8+ ...
Radeon RX 9070 GRE, AMD la porta in tutto il mondo | Recensione Gigabyte Gaming OC Radeon RX 9070 GRE, AMD la porta in tutto il mon...
Reolink OMVI 3i WiFi: videosorveglianza più intelligente e facile da usare Reolink OMVI 3i WiFi: videosorveglianza pi&ugrav...
TIM lancia il Pass Mondiali DAZN: 104 pa...
Tesla Roadster, promessa o miraggio? La ...
Mark Hamilton, la tavola periodica del m...
Hanger 13 annuncia Uomo d'Onore: espansi...
La battaglia delle HBM4 entra nel vivo: ...
Dopo 12 anni torna Alien: Isolation. Ecc...
ADATA Trusta ridurrà i costi di i...
SpaceX fornirà 110.000 GPU NVIDIA...
Hyundai IONIQ 6 N-Line, prova in antepri...
Sospesi i lavori di riparazione delle pe...
Formula V vi farà cambiare l'airf...
Netflix usa l'IA generativa per battere ...
Quando l'AI costruisce sé stessa:...
Meno ventole, più raffreddamento:...
Adidas Trionda: come funziona la tecnolo...
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: 10:06.


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