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 23-08-2010, 17:48   #1
Frank Lioty
Junior Member
 
Iscritto dal: Aug 2010
Messaggi: 6
[C] Liste generiche

Salve a tutti ragazzi (non c'è una sezione di presentazione?). Devo sviluppare un progetto per l'uni e devo scrivere delle funzioni che manipolino delle liste generiche da inserire poi in un opportuno header file.

I prototipi della funzione e le strutture della lista sono qui:

http://pastebin.com/w7HqhQbN

Le funzioni vengono poi "messe sotto stress" tramite un codice di test che cerca eventuali punti critici.

Ho implementato la New_List così:

Codice:
list_t * new_List(int (* compare) (void *, void *),void* (* copyk) (void *),void* (*copyp) (void*)) {

list_t *t;
t = (list_t*)malloc(sizeof(list_t));
if (t==NULL) return NULL;

t->head = NULL;
t->compare = compare;
t->copyk = copyk;
t->copyp = copyp;

return t;

}
le funzioni passate tramite chiamata di sistema sono fornite dal professore e permettono di costruire liste con elementi aventi per chiavi interi/stringhe ed informazioni (payload) interi/stringhe.

La mia free_List:

Codice:
void free_List (list_t ** pt) {

if(pt == NULL || *pt == NULL) return;
elem_t *curr = (*pt)->head;

while(curr != NULL)
{
elem_t *temp = curr;
free(curr->key);
free(curr->payload);
free(temp);
temp = (curr->next);
}
free(*pt);
*pt = NULL;

}
I problemi sorgono nel momento in cui nei test si prova a deallocare la lista in seguito ad inserimento di elementi, dandomi SIGABRT. Il debugger mi informa che l'istruzione che dà errore è quella in grassetto nella free_List. Da dire che la deallocazione di liste vuote invece non dà errori.

Altro particolare non da poco è che provando a visualizzare prima della chiamata alla free_List il contenuto della lista, questa non mi restituisce i key ed i payload dei vari elementi (ma solo puntatori a void).

Penso quindi che sia un problema di implementazione della funzione di aggiunta elemento che poi causa un errore nel momento in cui si invoca la free_List. Ecco il codice:

Codice:
int add_ListElement(list_t * t,void * key, void* payload) {

if (find_ListElement(t, key) != NULL) return -1;

elem_t *new_nodo = malloc(sizeof(elem_t));
if (new_nodo == NULL) return -1;

new_nodo->key = t->copyk(key);
new_nodo->payload = t->copyp(payload);
new_nodo->next = NULL;

new_nodo->next = t->head;
t->head = new_nodo;

return 0;

}
P.S: Invoco la find_ListElement perché non sono tollerati elementi con chiavi replicate

Qualcuno sa dirmi cosa non va? Ci sto sbattendo la testa da 4 giorni ormai...
Frank Lioty è offline   Rispondi citando il messaggio o parte di esso
Old 24-08-2010, 06:43   #2
sottovento
Senior Member
 
L'Avatar di sottovento
 
Iscritto dal: Nov 2005
Città: Texas
Messaggi: 1722
Sarebbe bello vedere tutto il codice, magari con un esempio pronto per essere compilato ed eseguito.
Ad ogni modo:

Codice:
void free_List (list_t ** pt) {

if(pt == NULL || *pt == NULL) return;
elem_t *curr = (*pt)->head;

while(curr != NULL)
{
elem_t *temp = curr;
free(curr->key);
free(curr->payload);
free(temp);
temp = (curr->next);
}
free(*pt);
*pt = NULL;

}
La variabile temp punta a curr. La deallochi

Codice:
free(temp)
e poi la usi
Codice:
temp = (curr->next);
__________________
In God we trust; all others bring data
sottovento è offline   Rispondi citando il messaggio o parte di esso
Old 24-08-2010, 11:22   #3
Frank Lioty
Junior Member
 
Iscritto dal: Aug 2010
Messaggi: 6
Cavolo hai ragione! Ho modificato in:

Codice:
    while (curr != NULL) {
	elem_t *temp = curr;
	free(curr->key);
	free(curr->payload);
	free(temp);
	curr = curr->next;
e ora SEMBRA andare...

Quote:
Sarebbe bello vedere tutto il codice, magari con un esempio pronto per essere compilato ed eseguito.
Queste sono le mie implementazioni delle funzioni:

http://pastebin.com/va2DP2H2

Mentre qui trovi il file di test:

http://pastebin.com/sa4xJuhh

I test andrebbero effettuati tramite una chiamata make ad un makefile fornito dalla prof, che però non riesco a far partire. Mi sono limitato a compilare e mandare in esecuzione tramite gdb ed il programma non dà errori. Poi non so se è la stessa cosa, anche se sinceramente dubito.
Frank Lioty è offline   Rispondi citando il messaggio o parte di esso
Old 24-08-2010, 11:38   #4
sottovento
Senior Member
 
L'Avatar di sottovento
 
Iscritto dal: Nov 2005
Città: Texas
Messaggi: 1722
Quote:
Originariamente inviato da Frank Lioty Guarda i messaggi
Cavolo hai ragione! Ho modificato in:

Codice:
    while (curr != NULL) {
	elem_t *temp = curr;
	free(curr->key);
	free(curr->payload);
	free(temp);
	curr = curr->next;
e ora SEMBRA andare...
Eh no, siamo ancora al punto di prima, perche'
Codice:
curr = curr->next;
viene eseguito su un pezzo di memoria che non esiste piu'.
Dovresti spostare questa operazione in modo da eseguirla subito: memorizzi in temp il valore di curr e, fintanto che la memoria e' valida, esegui la curr = curr->next. Poi puoi deallocare il tuo elemento senza problemi
__________________
In God we trust; all others bring data
sottovento è offline   Rispondi citando il messaggio o parte di esso
Old 24-08-2010, 13:23   #5
Frank Lioty
Junior Member
 
Iscritto dal: Aug 2010
Messaggi: 6
ok ho invertito le ultime 2 istruzioni. Visto che non ho ancora deallocato la memoria dedicata alla struct, ho ancora disponibili i puntatori all'elemento successivo, alla key e al payload (anche se questi due non puntano più ad aree di memoria visto che le ho deallocate). Quindi sposto il puntatore all'elemento successivo per poi deallocare il precedente, giusto?

Potresti dare un'occhiata al mio codice facendo partire i test? Con gdb dopo la fase di inserimento degli elementi ho messo un break per controllare che i key e i payload siano effettivamente stati "riempiti" correttamente facendo:

print lists->head->key

ma non restituisce alcun valore, semplicemente un puntatore void*, come se la memoria per il key venisse effettivamente allocata senza però inserirvi la stringa desiderata. Stesso discorso per i payload della testa e di tutti gli altri elementi...

Per la precisione l'output è questo:

Codice:
(gdb) print lists->head->key
$1 = (void *) 0x804c718
Frank Lioty è 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...
Il ritorno di Fumito Ueda, autore di Sha...
Cooler Master svela GPU Shield, la nuova...
Samsung Galaxy S27 Pro: sarà lui ...
Così Google ha ottimizzato Chrome...
Xiaomi non cambia idea: il display poste...
LG presenta in Italia le gamme TV Micro ...
Sette anni dopo l'annuncio, The Wolf Amo...
'Non avrete aumenti': la decisione shock...
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...
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: 15:22.


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