Torna indietro   Hardware Upgrade Forum > Software > Programmazione

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€
Il cuore digitale di F1 a Biggin Hill: l'infrastruttura Lenovo dietro la produzione media
Il cuore digitale di F1 a Biggin Hill: l'infrastruttura Lenovo dietro la produzione media
Nel Formula 1 Technology and Media Centre di Biggin Hill, la velocità delle monoposto si trasforma in dati, immagini e decisioni in tempo reale grazie all’infrastruttura Lenovo che gestisce centinaia di terabyte ogni weekend di gara e collega 820 milioni di spettatori nel mondo
DJI Osmo Mobile 8: lo stabilizzatore per smartphone con tracking multiplo e asta telescopica
DJI Osmo Mobile 8: lo stabilizzatore per smartphone con tracking multiplo e asta telescopica
Il nuovo gimbal mobile DJI evolve il concetto di tracciamento automatico con tre modalità diverse, un modulo multifunzionale con illuminazione integrata e controlli gestuali avanzati. Nel gimbal è anche presente un'asta telescopica da 215 mm con treppiede integrato, per un prodotto completo per content creator di ogni livello
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 28-07-2005, 20:33   #1
ri
Senior Member
 
L'Avatar di ri
 
Iscritto dal: Feb 2003
Città: fra casa e lavoro
Messaggi: 1061
[c++] costruttori, distruttori ed eccezioni

son cresciuto con questo dogma: "non scrivere mai un costruttore o un distruttore che possano sollevare eccezioni"

non mi sembra una cosa stupida: ad esempio se abbiamo un distruttore che esegue molte operazioni e la prima di queste solleva un'eccezione, che succede alle operazioni successive? non vengono eseguite, ma quindi si avrebbero dei leack di memoria (nell'ipotesi migliore...)

stessa cosa in un costruttore: la new può fallire ed è buona norma verificare il risultato di ogni new immediatamente dopo averla eseguita, ma qual è il comportamento migliore da tenere in questi casi? una throw all'interno del costruttore è contrario alla regoletta sopra, ma non posso sicuramente tenermi un oggetto "con dei problemi"

quindi mi sorge la domanda: è il dogma sbagliato (o troppo rigido) o mi sfugge qualcosa?
ri è offline   Rispondi citando il messaggio o parte di esso
Old 28-07-2005, 21:38   #2
fek
Senior Member
 
L'Avatar di fek
 
Iscritto dal: Oct 2002
Città: San Jose, California
Messaggi: 11794
Quote:
Originariamente inviato da ri
quindi mi sorge la domanda: è il dogma sbagliato (o troppo rigido) o mi sfugge qualcosa?
Il dogma cosi' come lo hai enunciato e' parzialmente errato (o parzialmente giusto).

E' sacrosanto non sollevare mai un'eccezione in un distruttore.
I motivi sono vari, uno dei piu' importanti e':

- quando viene sollevata un'eccezione, per Standard C++, gli oggetti locali sullo stack dal punto in cui viene sollevata l'eccezione al punto in cui viene gestita sono distrutti invocando il distruttore (Stack Unwinding); che succede se uno di questi distruttori solleva a sua volta un'eccezione?

Si scatena il panico e sei nella terra dell"Undefined Behavior". Di solito i compilatori C++ chiamano immediatamente terminate() perche' non c'e' modo di uscire da quella situazione. Ma lo Standard non lo impone ed ogni compilatore e' libero di fare quello che gli pare.

Il discorso sul costruttore invece e' diverso; e' buona norma lanciare un'eccezione da un costruttore per riportare una situazione d'errore che ti impedisce di costruire l'oggetto in maniera corretta; ed e' una situazione che accade piuttosto spesso.
Ci sono vari modi per risolverla, ma nessuno e' elegante e semplice da implementare come sollevare un'eccezione, perche' lo Standard C++ in questo caso ti viene in aiuto, imponendo che in caso di eccezione durante la creazioen di un'oggetto dinamico, l'operatore new ritorna NULL.

Esempio:

Codice:
A* a;

try 
{
  a = new A();
}
catch (...)      // non fatelo mai, indicate sempre l'eccezione di cui fare il trap
{
   // a qui e' uguale a NULL perche' e' stata sollevata un'eccezione 
   // nel costruttore e qualcosa e' andato storto
}
 
if (a == NULL)  // E quanto ci starebbe bene qui un NULL Object Pattern?
{
   printf("AAAAAAAAAAAARGH!!!");
}
Senza eccezioni la soluziona sarebbe stata un po' meno elegante:

Codice:
A* a = new A();

// Devo creare un metodo IsValid() solo per riportare la condizione d'errore

if (!a->IsValid())  
{
   // E se dimentico di distruggere l'oggetto esplicitamente? LEAK!
   delete a;
   a = 0;
}

if (a == 0)
{
  printf("AAAAAAAAAAAAARGH");
}
fek è offline   Rispondi citando il messaggio o parte di esso
Old 28-07-2005, 22:18   #3
ri
Senior Member
 
L'Avatar di ri
 
Iscritto dal: Feb 2003
Città: fra casa e lavoro
Messaggi: 1061
ottima spiegazione, conferma quanto avevo intuito, ty
ri è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 14:36   #4
RaouL_BennetH
Senior Member
 
L'Avatar di RaouL_BennetH
 
Iscritto dal: Sep 2004
Messaggi: 3967
piccola curiosità su NULL:

Sono alle pagine iniziali del libro di Stroustroup (o come accidenti si scrive ) e lui dice che in genere non è una buona idea assegnare a qualcosa il valore NULL ma si dovrebbe preferire il valore zero. E' appunto un suggerimento delle pagine iniziali che poi viene approfondito, o in generale in C++ si fa così ?

Thx.

RaouL.
__________________
Dai wafer di silicio nasce: LoHacker... il primo biscotto Geek
RaouL_BennetH è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 14:40   #5
ilsensine
Senior Member
 
L'Avatar di ilsensine
 
Iscritto dal: Apr 2000
Città: Roma
Messaggi: 15625
Quote:
Originariamente inviato da RaouL_BennetH
piccola curiosità su NULL:

Sono alle pagine iniziali del libro di Stroustroup (o come accidenti si scrive ) e lui dice che in genere non è una buona idea assegnare a qualcosa il valore NULL ma si dovrebbe preferire il valore zero. E' appunto un suggerimento delle pagine iniziali che poi viene approfondito, o in generale in C++ si fa così ?
Dosemu "mappa" memoria perfettamente valida all'indirizzo 0.

Su alcuni sistemi, l'indirizzo "0" è perfettamente valido.
__________________
0: or %edi, %ecx; adc %eax, (%edx); popf; je 0b-22; pop %ebx; fadds 0x56(%ecx); lds 0x56(%ebx), %esp; mov %al, %al
andeqs pc, r1, #147456; blpl 0xff8dd280; ldrgtb r4, [r6, #-472]; addgt r5, r8, r3, ror #12
ilsensine è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 18:44   #6
fek
Senior Member
 
L'Avatar di fek
 
Iscritto dal: Oct 2002
Città: San Jose, California
Messaggi: 11794
Quote:
Originariamente inviato da RaouL_BennetH
piccola curiosità su NULL:

Sono alle pagine iniziali del libro di Stroustroup (o come accidenti si scrive ) e lui dice che in genere non è una buona idea assegnare a qualcosa il valore NULL ma si dovrebbe preferire il valore zero. E' appunto un suggerimento delle pagine iniziali che poi viene approfondito, o in generale in C++ si fa così ?

Thx.

RaouL.
Io uso 0. Voi?
fek è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 19:29   #7
ri
Senior Member
 
L'Avatar di ri
 
Iscritto dal: Feb 2003
Città: fra casa e lavoro
Messaggi: 1061
NULL, trovo che metta in evidenza il fatto che si sta lavorando con un puntatore e non con una variabile normale

e casomai dovessi ricompilare per una piattaforma in cui il NULL non è zero ci vuol poco a ridefinirlo, andare a cambiare gli 0 no
ri è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 19:46   #8
ghiotto86
Senior Member
 
L'Avatar di ghiotto86
 
Iscritto dal: Jul 2004
Città: Napoli
Messaggi: 2029
Quote:
Originariamente inviato da fek
Io uso 0. Voi?
io 0 (zero).
è preferibile usare 0 perchè non è detto che NULL sia definito
ghiotto86 è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 21:15   #9
maxithron
Senior Member
 
L'Avatar di maxithron
 
Iscritto dal: Mar 2002
Città: Italy/Usa
Messaggi: 2817
in c++ io pure(intendo 0), anche per seguitare a limitare l'uso di macro.
__________________
"Utilizzando atomi pentavalenti drogheremo il silicio di tipo n;
Utilizzando atomi trivalenti drogheremo il silicio di tipo p;
Utilizzando della cannabis ci drogheremo noi e vedremo il silicio fare cose impossibili" - DSDT-HowTo

Ultima modifica di maxithron : 29-07-2005 alle 21:19.
maxithron è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 21:20   #10
end.is.forever
Senior Member
 
Iscritto dal: Jul 2004
Messaggi: 1578
E' buona abitudine eseguire le operazioni che possono scatenare eccezioni in un metodo di inizializzazione invece che nel costruttore, e nel metodo di finalizzazione invece che nel distruttore.

Lo scopo di distruttore e costruttore deve limitarsi all'allocazione e alla liberazione della memoria o poco più.

Stessa cosa per l'acquisizione o il rilascio di risorse, e per le operazioni che possono comportare sospensioni.
end.is.forever è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 21:43   #11
ri
Senior Member
 
L'Avatar di ri
 
Iscritto dal: Feb 2003
Città: fra casa e lavoro
Messaggi: 1061
se il metodo di finalizzazione richiamato dal distruttore solleva eccezioni sei comunque nella merda
ri è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 21:56   #12
end.is.forever
Senior Member
 
Iscritto dal: Jul 2004
Messaggi: 1578
Quote:
Originariamente inviato da ri
se il metodo di finalizzazione richiamato dal distruttore solleva eccezioni sei comunque nella merda
No perchè il metodo di inizializzazione lo chiami dopo il costruttore e il metodo di finalizzazione prima del distruttore.

Ti faccio un esempio, ho una classe che gestisce la connessione con un certo database.
Uno potrebbe pensare di aprire la connessione (con eventuali eccezioni) nel costruttore, e chiuderla nel distruttore.

Molto meglio invece inserire nel costruttore solo l'assegnamento dei campi a seconda dei parametri passati; poi, esplicitamente alla chiamata del metodo di inizializzazione, aprire la connessione. Se si scatena lì l'eccezione è tutto un' altro paio di maniche rispetto al costruttore.
Stessa cosa per distruttore.
end.is.forever è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 22:55   #13
fek
Senior Member
 
L'Avatar di fek
 
Iscritto dal: Oct 2002
Città: San Jose, California
Messaggi: 11794
Quote:
Originariamente inviato da end.is.forever
Molto meglio invece inserire nel costruttore solo l'assegnamento dei campi a seconda dei parametri passati; poi, esplicitamente alla chiamata del metodo di inizializzazione, aprire la connessione. Se si scatena lì l'eccezione è tutto un' altro paio di maniche rispetto al costruttore.
Stessa cosa per distruttore.
Non sono d'accorso sul secondo metodo di inizializzazione, perche' non e' altro che una duplicazione di un concetto che gia' esiste (il costruttore). Perche' matenere due metodi (costruttore e metodo di inizializzazione) quando se ne puo' mantenere solo uno (costruttore)?

Ed inoltre usando un metodo di inizializzazione si perde l'automatismo garantito dallo Standard che distrugge automaticamente l'oggetto in caso di eccezione. Con un metodo di inizializzazione, il programmatore deve ricordare di distruggere l'oggetto ed una riga di codice in piu' siginifica una riga in piu' che magari ci si dimentica di scrivere oppure che puo' introdurre un bug

Inoltre, per definizione il costruttore dovrebbe lasciare l'oggetto in uno stato "valido", mentre imporre a cliente dell'oggetto di ricordare di chiamare un'ulteriore metodo per la concludere la costruzione significa imporre un ulteriore complicazione al cliente, complicazione che puo' introdurre un bug (il cliente magari lo dimentica).

Meglio semplificare.
fek è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 22:59   #14
cionci
Senior Member
 
L'Avatar di cionci
 
Iscritto dal: Apr 2000
Città: Vicino a Montecatini(Pistoia) Moto:Kawasaki Ninja ZX-9R Scudetti: 29
Messaggi: 53971
In pratica end.is.forever vuole dire che secondo lui conviene cerare un metodo di finalizzazione da chiamare esplicitamente prima che la classe venga distrutta...

Io aggiungo che è comodo se l'oggetto viene istanziato solo come membro di una classe e non come variabile locale ad un metodo (altrimenti bisogna ricordarsi di finalizzare l'oggetto prima della distruzione automatica), ovviamente IMHO...
cionci è offline   Rispondi citando il messaggio o parte di esso
Old 29-07-2005, 23:00   #15
fek
Senior Member
 
L'Avatar di fek
 
Iscritto dal: Oct 2002
Città: San Jose, California
Messaggi: 11794
Quote:
Originariamente inviato da cionci
In pratica end.is.forever vuole dire che secondo lui conviene cerare un metodo di finalizzazione da chiamare esplicitamente prima che la classe venga distrutta...

Io aggiungo che è comodo se l'oggetto viene istanziato solo come membro di una classe e non come variabile locale ad un metodo (altrimenti bisogna ricordarsi di finalizzare l'oggetto prima della distruzione automatica), ovviamente IMHO...
Il metodo di finalizzazione che lancia eventuali eccezioni e' sicuramente una buona idea per ovviare al problema di non poter lanciare eccezioni nel distruttore.
fek è offline   Rispondi citando il messaggio o parte di esso
Old 30-07-2005, 09:32   #16
end.is.forever
Senior Member
 
Iscritto dal: Jul 2004
Messaggi: 1578
Ovviamente io parlo in teoria, non voglio dire che per ogni classe si debba fare questo.
Io esplicitamente lo faccio poche volte, ma è un pattern che si usa implicitamente spessissimo, quando si vogliono separare le fasi di istanziazione e distruzione da quella di utilizzo.

Qualche esempio: eventi load e close di una finestra, apertura e chiusura di una connessione, acquisizione e rilascio di risorse...
end.is.forever è offline   Rispondi citando il messaggio o parte di esso
Old 30-07-2005, 10:08   #17
fek
Senior Member
 
L'Avatar di fek
 
Iscritto dal: Oct 2002
Città: San Jose, California
Messaggi: 11794
Quote:
Originariamente inviato da end.is.forever
Ovviamente io parlo in teoria, non voglio dire che per ogni classe si debba fare questo.
Io esplicitamente lo faccio poche volte, ma è un pattern che si usa implicitamente spessissimo, quando si vogliono separare le fasi di istanziazione e distruzione da quella di utilizzo.

Qualche esempio: eventi load e close di una finestra, apertura e chiusura di una connessione, acquisizione e rilascio di risorse...
Io parlo sia in teoria sia in pratica: separare costruttore e inizializzazione non e' una buona idea se non c'e' un motivo davvero ottimo per farlo, perche' rende l'interfaccia piu' complessa.
fek è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


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...
Recensione Pura 80 Pro: HUAWEI torna a stupire con foto spettacolari e ricarica superveloce Recensione Pura 80 Pro: HUAWEI torna a stupire c...
Opera Neon: il browser AI agentico di nuova generazione Opera Neon: il browser AI agentico di nuova gene...
Snap e Perplexity unite: dal prossimo an...
La Cina dice addio a NVIDIA? Il governo ...
Microlino, simbolo italiano della mobili...
Apple disattiverà la sincronizzaz...
Google lancia l'allarme: attenzione ai m...
Primo test drive con Leapmotor B10: le c...
'Non può essere un robot': l'uman...
Monopattino elettrico Segway Ninebot Max...
Syberia Remastered è disponibile:...
Sony scopre che tutti i modelli AI hanno...
Amazon nasconde un -15% su 'Seconda Mano...
Due occasioni Apple su Amazon: iPhone 16...
Verso la fine della TV tradizionale? I g...
Cassa JBL a 39€, portatili, smartphone, ...
Cometa interstellare 3I/ATLAS: la sonda ...
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: 02:37.


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