Piojolopez2406
02-06-2007, 12:59
Salve ragazzi ho fatto un programmino che gestisce una lista lineare senza nodo sentinella, ora vorrei inserire il nodo sentinella così da poter eliminare 1 function di inserimento ed 1 di eliminazione, ma non riesco propria capire cosa devo fare per creare sto nodo sentinella, mi dareste una mano...???
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
/* Strutture autoriferente dinamica */
typedef struct {
char nome[20];
short age;
} INFO_FIELD;
/* Prototipi di funzione */
char *crea_lista();
void insert_testa(short, INFO_FIELD *, void **);
void insert_nodo(short, INFO_FIELD *, void **);
void delete_testa(void **);
void delete_nodo(void **);
INFO_FIELD legge_infonodo(struct PERSONA **);
void visualizza_lista(void **);
void messaggio_errore();
void main()
{
struct PERSONA {
INFO_FIELD info;
struct PERSONA *p_next;
};
struct PERSONA *head, *punt;
short scelta, n;
INFO_FIELD dato;
head = (struct PERSONA *) crea_lista();
do {
printf("QUESTO PROGRAMMA GESTISCE UNA LISTA GENERICA\n\n");
puts("[1] Aggiungi elemento nella lista");
puts("[2] Elimina elemento dalla lista");
puts("[3] Visualizza lista");
puts("[0] Esci dal programma");
printf("\nscelta: ");
scanf("%hd", &scelta);
switch(scelta) {
case 1:
printf("\n\nA G G I U N G I E L E M E N T O\n\n");
fflush(stdin);
printf("Inserire nome: ");
gets(dato.nome);
printf("Inserire eta': ");
scanf("%hd", &dato.age);
do {
puts("\n[ 1 ] Inserimento in testa alla lista");
puts("[ 2 ] Inserimento dopo nodo corrente (solo se la lista non e' vuota)");
printf("\nscelta: ");
scanf("%hd", &n);
} while (!(scelta == 1 || scelta == 2));
if(n == 1) {
insert_testa(sizeof(dato), &dato, &head);
punt = head;
}
else
insert_nodo(sizeof(dato), &dato, &punt);
printf("\n\n");
break;
case 2:
printf("\n\nE L I M I N A E L E M E N T O\n\n");
do {
puts("\n[1] Elimina nodo di testa");
puts("[2] Elimina nodo successore al nodo corrente");
printf("\nscelta: ");
scanf("%hd", &n);
switch(n) {
case 1:
delete_testa(&head);
break;
case 2:
delete_nodo(&punt);
break;
default:;
}
} while (!(scelta == 1 || scelta == 2));
case 3:
printf("\n\nVISUALIZZA LISTA\n\n");
visualizza_lista(&head);
printf("\n\n");
break;
}
} while(scelta != 0);
}
/*
* Questa function inizializza il nodo di testa della la lista lineare
*/
char *crea_lista()
{
char *testa;
testa = NULL;
return testa;
}
/*
* Questa function inserisce un nodo in testa ad una lista lineare.
* Prende in input la dimensione del elemento da inserire, l'elemento
* e il puntatore al nodo di testa.
* Definisce una struttura lista e con la funzione 'memcpy' copia
* nel nodo 'ptr' (appena creato) l'elemento dato di dimensione len_info,
* a questo punto fa puntare il nodo creato al nodo di testa e assegna
* al nodo appena creato il puntatore 'p_head'.
*/
void insert_testa(short len_info, INFO_FIELD *dato, void **p_head)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *ptr;
ptr = calloc(1, sizeof(struct lista));
memcpy(ptr, dato, len_info);
ptr->p_next = (struct lista *) *p_head;
(struct lista *) *p_head = ptr;
}
/*
* Questa function inserisce un nodo dopo il nodo corrente di una lista
* lineare. Prende in input la dimensione del dato, il dato da inserire
* nel nodo creato e il punatatore al nodo corrente.
* Con la funzione memcpy, copia il contenuto di dato nel nodo appena creayto
* creato. Fa diventare successore del nodo creato, il nodo successore
* del nodo corrente, fa diventare il nodo creato successore
* del nodo corrente. Fa puntare il puntatore al nodo corrente al nodo
* appena creato.
* Se si tenta di aggiungere un elemento dopo il nodo corrente quand la lista
* è vuota si originerà un errore di windows nell' esecuzione.
*/
void insert_nodo(short len_info, INFO_FIELD *dato, void **p_punt)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *ptr;
ptr = calloc(1, sizeof(struct lista));
if(ptr == NULL)
messaggio_errore();
memcpy(ptr, dato, len_info);
ptr->p_next = ((struct lista *)*p_punt)->p_next;
((struct lista *)*p_punt)->p_next = ptr;
(struct lista *)*p_punt = ptr;
}
/*
* Questa function elimina il nodo di testa in una lista lineare.
* Prende in input il puntatore alla testa della lista, controlla che la
* lista non sia vuota, e in caso negativo inserisce nel nodo 'temp'
* il successore del nodo di testa, libera il nodo di testa e fa puntare
* dal puntatore p_head il nodo temp.
*/
void delete_testa(void **p_head)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *temp;
if(!(struct lista *)*p_head) {
printf("\n\n\n *** L I S T A V U O T A ***\n\n\n");
return;
}
temp = ((struct lista *)*p_head)->p_next;
free((struct lista *)*p_head);
*p_head = temp;
}
/*
* Questa function elimina il nodo corrente in una lista lineare.
* Prende in input il nodo corrente, controlla se la lista non è
* vuota e se il nodo corrente ha un successore.
* Se la lista non è vuota e il nodo corrente ha un successore,
* fa puntare da ptr il nodo successore del nodo corrente, fa diventare
* il successore di 'ptr', successore del nodo corrente, elimina il nodo
* puntato da ptr.
*/
void delete_nodo(void **p_punt)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *ptr;
if(!(struct lista *)*p_punt) {
printf("\n\n\n *** L I S T A V U O T A ***\n\n\n");
return;
} else if(!((struct lista *)*p_punt)->p_next) {
printf("\n\n\n *** NON ESISTE SUCCESSORE ***\n\n\n");
return;
}
ptr = ((struct lista *)*p_punt)->p_next;
((struct lista *)*p_punt)->p_next = ptr->p_next;
free(ptr);
}
//Questa function visualizza a schermo la lista lineare
void visualizza_lista(void **p_head)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *temp;
temp = (struct lista *)*p_head;
while (temp != NULL){
printf("[%s, %d] -> ", temp->info.nome, temp->info.age);
temp = temp->p_next;
}
printf("NULL\n\n");
}
// Questa function visualizza un messaggio di errore.
void messaggio_errore()
{
printf("Memoria insufficiente\n");
exit(1);
}
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
/* Strutture autoriferente dinamica */
typedef struct {
char nome[20];
short age;
} INFO_FIELD;
/* Prototipi di funzione */
char *crea_lista();
void insert_testa(short, INFO_FIELD *, void **);
void insert_nodo(short, INFO_FIELD *, void **);
void delete_testa(void **);
void delete_nodo(void **);
INFO_FIELD legge_infonodo(struct PERSONA **);
void visualizza_lista(void **);
void messaggio_errore();
void main()
{
struct PERSONA {
INFO_FIELD info;
struct PERSONA *p_next;
};
struct PERSONA *head, *punt;
short scelta, n;
INFO_FIELD dato;
head = (struct PERSONA *) crea_lista();
do {
printf("QUESTO PROGRAMMA GESTISCE UNA LISTA GENERICA\n\n");
puts("[1] Aggiungi elemento nella lista");
puts("[2] Elimina elemento dalla lista");
puts("[3] Visualizza lista");
puts("[0] Esci dal programma");
printf("\nscelta: ");
scanf("%hd", &scelta);
switch(scelta) {
case 1:
printf("\n\nA G G I U N G I E L E M E N T O\n\n");
fflush(stdin);
printf("Inserire nome: ");
gets(dato.nome);
printf("Inserire eta': ");
scanf("%hd", &dato.age);
do {
puts("\n[ 1 ] Inserimento in testa alla lista");
puts("[ 2 ] Inserimento dopo nodo corrente (solo se la lista non e' vuota)");
printf("\nscelta: ");
scanf("%hd", &n);
} while (!(scelta == 1 || scelta == 2));
if(n == 1) {
insert_testa(sizeof(dato), &dato, &head);
punt = head;
}
else
insert_nodo(sizeof(dato), &dato, &punt);
printf("\n\n");
break;
case 2:
printf("\n\nE L I M I N A E L E M E N T O\n\n");
do {
puts("\n[1] Elimina nodo di testa");
puts("[2] Elimina nodo successore al nodo corrente");
printf("\nscelta: ");
scanf("%hd", &n);
switch(n) {
case 1:
delete_testa(&head);
break;
case 2:
delete_nodo(&punt);
break;
default:;
}
} while (!(scelta == 1 || scelta == 2));
case 3:
printf("\n\nVISUALIZZA LISTA\n\n");
visualizza_lista(&head);
printf("\n\n");
break;
}
} while(scelta != 0);
}
/*
* Questa function inizializza il nodo di testa della la lista lineare
*/
char *crea_lista()
{
char *testa;
testa = NULL;
return testa;
}
/*
* Questa function inserisce un nodo in testa ad una lista lineare.
* Prende in input la dimensione del elemento da inserire, l'elemento
* e il puntatore al nodo di testa.
* Definisce una struttura lista e con la funzione 'memcpy' copia
* nel nodo 'ptr' (appena creato) l'elemento dato di dimensione len_info,
* a questo punto fa puntare il nodo creato al nodo di testa e assegna
* al nodo appena creato il puntatore 'p_head'.
*/
void insert_testa(short len_info, INFO_FIELD *dato, void **p_head)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *ptr;
ptr = calloc(1, sizeof(struct lista));
memcpy(ptr, dato, len_info);
ptr->p_next = (struct lista *) *p_head;
(struct lista *) *p_head = ptr;
}
/*
* Questa function inserisce un nodo dopo il nodo corrente di una lista
* lineare. Prende in input la dimensione del dato, il dato da inserire
* nel nodo creato e il punatatore al nodo corrente.
* Con la funzione memcpy, copia il contenuto di dato nel nodo appena creayto
* creato. Fa diventare successore del nodo creato, il nodo successore
* del nodo corrente, fa diventare il nodo creato successore
* del nodo corrente. Fa puntare il puntatore al nodo corrente al nodo
* appena creato.
* Se si tenta di aggiungere un elemento dopo il nodo corrente quand la lista
* è vuota si originerà un errore di windows nell' esecuzione.
*/
void insert_nodo(short len_info, INFO_FIELD *dato, void **p_punt)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *ptr;
ptr = calloc(1, sizeof(struct lista));
if(ptr == NULL)
messaggio_errore();
memcpy(ptr, dato, len_info);
ptr->p_next = ((struct lista *)*p_punt)->p_next;
((struct lista *)*p_punt)->p_next = ptr;
(struct lista *)*p_punt = ptr;
}
/*
* Questa function elimina il nodo di testa in una lista lineare.
* Prende in input il puntatore alla testa della lista, controlla che la
* lista non sia vuota, e in caso negativo inserisce nel nodo 'temp'
* il successore del nodo di testa, libera il nodo di testa e fa puntare
* dal puntatore p_head il nodo temp.
*/
void delete_testa(void **p_head)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *temp;
if(!(struct lista *)*p_head) {
printf("\n\n\n *** L I S T A V U O T A ***\n\n\n");
return;
}
temp = ((struct lista *)*p_head)->p_next;
free((struct lista *)*p_head);
*p_head = temp;
}
/*
* Questa function elimina il nodo corrente in una lista lineare.
* Prende in input il nodo corrente, controlla se la lista non è
* vuota e se il nodo corrente ha un successore.
* Se la lista non è vuota e il nodo corrente ha un successore,
* fa puntare da ptr il nodo successore del nodo corrente, fa diventare
* il successore di 'ptr', successore del nodo corrente, elimina il nodo
* puntato da ptr.
*/
void delete_nodo(void **p_punt)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *ptr;
if(!(struct lista *)*p_punt) {
printf("\n\n\n *** L I S T A V U O T A ***\n\n\n");
return;
} else if(!((struct lista *)*p_punt)->p_next) {
printf("\n\n\n *** NON ESISTE SUCCESSORE ***\n\n\n");
return;
}
ptr = ((struct lista *)*p_punt)->p_next;
((struct lista *)*p_punt)->p_next = ptr->p_next;
free(ptr);
}
//Questa function visualizza a schermo la lista lineare
void visualizza_lista(void **p_head)
{
struct lista {
INFO_FIELD info;
struct lista *p_next;
} *temp;
temp = (struct lista *)*p_head;
while (temp != NULL){
printf("[%s, %d] -> ", temp->info.nome, temp->info.age);
temp = temp->p_next;
}
printf("NULL\n\n");
}
// Questa function visualizza un messaggio di errore.
void messaggio_errore()
{
printf("Memoria insufficiente\n");
exit(1);
}