Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Recensione vivo X300 Pro: è ancora lui il re della fotografia mobile, peccato per la batteria
Recensione vivo X300 Pro: è ancora lui il re della fotografia mobile, peccato per la batteria
vivo X300 Pro rappresenta un'evoluzione misurata della serie fotografica del produttore cinese, con un sistema di fotocamere migliorato, chipset Dimensity 9500 di ultima generazione e l'arrivo dell'interfaccia OriginOS 6 anche sui modelli internazionali. La scelta di limitare la batteria a 5.440mAh nel mercato europeo, rispetto ai 6.510mAh disponibili altrove, fa storcere un po' il naso
Lenovo Legion Go 2: Ryzen Z2 Extreme e OLED 8,8'' per spingere gli handheld gaming PC al massimo
Lenovo Legion Go 2: Ryzen Z2 Extreme e OLED 8,8'' per spingere gli handheld gaming PC al massimo
Lenovo Legion Go 2 è la nuova handheld PC gaming con processore AMD Ryzen Z2 Extreme (8 core Zen 5/5c, GPU RDNA 3.5 16 CU) e schermo OLED 8,8" 1920x1200 144Hz. È dotata anche di controller rimovibili TrueStrike con joystick Hall effect e una batteria da 74Wh. Rispetto al dispositivo che l'ha preceduta, migliora ergonomia e prestazioni a basse risoluzioni, ma pesa 920g e costa 1.299€ nella configurazione con 32GB RAM/1TB SSD e Z2 Extreme
AWS re:Invent 2025: inizia l'era dell'AI-as-a-Service con al centro gli agenti
AWS re:Invent 2025: inizia l'era dell'AI-as-a-Service con al centro gli agenti
A re:Invent 2025, AWS mostra un’evoluzione profonda della propria strategia: l’IA diventa una piattaforma di servizi sempre più pronta all’uso, con agenti e modelli preconfigurati che accelerano lo sviluppo, mentre il cloud resta la base imprescindibile per governare dati, complessità e lock-in in uno scenario sempre più orientato all’hybrid cloud
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


Recensione vivo X300 Pro: è ancora lui il re della fotografia mobile, peccato per la batteria Recensione vivo X300 Pro: è ancora lui il...
Lenovo Legion Go 2: Ryzen Z2 Extreme e OLED 8,8'' per spingere gli handheld gaming PC al massimo Lenovo Legion Go 2: Ryzen Z2 Extreme e OLED 8,8'...
AWS re:Invent 2025: inizia l'era dell'AI-as-a-Service con al centro gli agenti AWS re:Invent 2025: inizia l'era dell'AI-as-a-Se...
Cos'è la bolla dell'IA e perché se ne parla Cos'è la bolla dell'IA e perché se...
BOOX Palma 2 Pro in prova: l'e-reader diventa a colori, e davvero tascabile BOOX Palma 2 Pro in prova: l'e-reader diventa a ...
Roborock Q10 S5+ a un prezzo molto conve...
Upgrade PC a prezzo ridotto: le migliori...
Sono i 6 smartphone migliori su Amazon: ...
Google Pixel 9a a 361€, mai così ...
Super sconti sugli spazzolini Oral-B, an...
Aspira a 6000Pa, lava bene, costa 139€: ...
Nuove scorte: torna il portatile tuttofa...
Toyota usa giochi e premi per spingere i...
HarmonyOS ha raggiunto la soglia di sopr...
Le offerte Amazon più convenienti...
Un gruppo di ladri ha usato Google Maps ...
Apple non si fida di Samsung per la real...
Windows 11: un nuovo driver nativo mette...
Vi hanno regalato buoni Amazon? Intanto ...
Via acari, polvere e sporco da materassi...
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: 09:12.


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