|
|||||||
|
|
|
![]() |
|
|
Strumenti |
|
|
#1 |
|
Member
Iscritto dal: Sep 2001
Messaggi: 163
|
[C++]Strano comportamento costruttore di copia ridefinito
Salve a tutti, vi posto quà di seguito il codice di un programmino d'esempio per spiegare il mio problema:
Codice:
#include <cstdlib>
#include <iostream>
#include <math.h>
using namespace std;
struct punto {double x, y;};
class poligono
{ int nvertici; punto* pp;
public:
poligono();
poligono (int n, const punto v[]);
~poligono();
poligono(const poligono& pol);
};
poligono::poligono()
{
nvertici = 1; pp = new punto[nvertici];
}
poligono::poligono(int n, const punto v[])
{
nvertici = n;
pp = new punto[nvertici];
for (int i=0; i<nvertici; i++)
{
(pp+i)->x = v[i].x;
(pp+i)->y = v[i].y;
}
}
poligono::~poligono()
{
delete[] pp;
}
poligono::poligono(const poligono& pol)
{
cout<< "costruttore di copia\n";
nvertici = pol.nvertici;
pp = new punto[nvertici];
for (int i= 0; i < nvertici; i++)
{
(pp+i)->x = (pol.pp+i)->x;
(pp+i)->y = (pol.pp+i)->y;
}
}
int main()
{
punto cc[3];
poligono bz();
poligono test = bz; // errore a tempo di compilazione "conversion from poligono()() to non-scalar type poligono requested
poligono test2 = poligono(3, cc);
system("PAUSE");
return EXIT_SUCCESS;
}
Nel main ho aggiunto i commenti alle righe che non mi quadrano, innanzitutto l'inizializzazione dell'ogetto "test" di tipo "poligono" con l'ogetto "bz" sempre di tipo "poligono",non capisco proprio perchè mi esca fuori quell'errore quando compilo. Secondo cosa che non mi spiego, perchè quando inizializzo l'ogetto di tipo "poligono" test2 con il costruttore "poligono(3, cc)" non viene richamato il costruttore di copia? Da quello che so in quel caso dovrebbe intervenire proprio il costruttore di copia che ho ridefinito, però se mando in esecuzione l'exe non mi esce il cout del costruttore di copia. Sapreste darmi qualche dritta?
__________________
CASE:NZXT S340 Midi-Tower Nero - Rosso, ALI:EVGA SuperNOVA 650 G2 650W, 80 PLUS Gold, Full modular, MOBO:MSI H270 GAMING M3, CPU:Intel Core i7-7700, DISSY:Noctua NH-U12S, RAM:Corsair Vengeance LPX Red, 2x8GB, 2400MHz DDR4, CL16, VGA:ZOTAC GeForce GTX 1080 AMP, HD:SEAGATE Barracuda 7200.11 1000GB + Crucial SSD MX300 525GB SATA3 |
|
|
|
|
|
#2 |
|
Member
Iscritto dal: Aug 2003
Messaggi: 125
|
Temo che "poligono bz();" venga interpretato come un prototipo di
funzione, quindi la sintassi corretta nel tuo caso e' "poligono bz;". Per la seconda domanda, sinceramente non ricordo, quindi potrei andare ad intuito, cioe' che "poligono test2 = poligono(3, cc);" equivalga a "poligono test2(3, cc);". Ma a questo immagino ci eri gia' arrivato da solo... =) Ultima modifica di Bane : 04-12-2005 alle 11:32. |
|
|
|
|
|
#3 |
|
Senior Member
Iscritto dal: Apr 2003
Città: Genova
Messaggi: 4747
|
da C++ Lessons:
Il costruttore di copia è uno speciale costruttore che prende come argomento un reference a un oggetto della stessa classe e crea un nuovo oggetto che ne è una copia. Per default, il compilatore mette a disposizione un costruttore di copia che effettua una copia membro a membro dall' oggetto originale a quello che viene creato (cosiddetta copia member-wise)... nel tuo caso, la riga poligono test2 = poligono(3, cc); non dovrebbe chiamare mai il costruttore di copia, e perchè cc è di tipo punto[], e perchè i parametri passati corrispondono agli argomenti del costruttore PS: di solito (per come mi hanno insegnato) è consigliabile implementare un default (NULL) constructor e ridefinire l' operatore di assegnazione...
__________________
Jappilas is a character created by a friend for his own comic - I feel honored he allowed me to bear his name Saber's true name belongs to myth - a Heroic Soul out of legends, fighting in our time to fullfill her only wish Let her image remind of her story, and of the emotions that flew from my heart when i assisted to her Fate
Ultima modifica di jappilas : 04-12-2005 alle 13:08. |
|
|
|
|
|
#4 |
|
Member
Iscritto dal: Apr 2004
Messaggi: 130
|
In questo caso:
Codice:
poligono test2 = poligono(3, cc); Pero', visto che viene creato un oggetto temporaneo utilizzato dal copy-constructor, al compilatore e' permesso saltare quest'ultimo e creare direttamente l'oggetto. Praticamente, come ha detto Bane, e' come fosse scritto: Codice:
poligono test2(3, cc); |
|
|
|
|
|
#5 |
|
Member
Iscritto dal: Sep 2001
Messaggi: 163
|
Secondo il libro del mio prof con quell'inizializzazione dovrebbe essere richiamato proprio il costruttore di copia.
A questo punto, visto che sto esaurendo con questo argomento vi riporto un altro piccolo pezzo di codice: Codice:
#include <cstdlib>
#include <iostream>
#include <string.h>
using namespace std;
class stringa
{ char* str;
public:
stringa(const char s[]);
~stringa();
};
stringa::stringa(const char s[])
{
str = new char[strlen(s)+1];
strcpy(str, s);
cout << "ogetto costruito \n";
}
stringa::~stringa()
{
delete[] str;
cout << "ogetto distrutto \n";
}
stringa ff()
{
stringa sg("acqua");
return sg;
}
int main()
{
stringa sa = ff();
system("PAUSE");
return EXIT_SUCCESS;
}
Io ho provato a compilarlo e va tutto bene, alla fine sa risulta proprio inizializzato con il valore restituito da ff, e non solo, il distruttore dell'oggetto sg non viene mai richiamato. Dove sta la verità?
__________________
CASE:NZXT S340 Midi-Tower Nero - Rosso, ALI:EVGA SuperNOVA 650 G2 650W, 80 PLUS Gold, Full modular, MOBO:MSI H270 GAMING M3, CPU:Intel Core i7-7700, DISSY:Noctua NH-U12S, RAM:Corsair Vengeance LPX Red, 2x8GB, 2400MHz DDR4, CL16, VGA:ZOTAC GeForce GTX 1080 AMP, HD:SEAGATE Barracuda 7200.11 1000GB + Crucial SSD MX300 525GB SATA3 |
|
|
|
|
|
#6 |
|
Member
Iscritto dal: Aug 2003
Messaggi: 125
|
Vediamo, cosi' ad occhio dovrebbe succedere quanto segue:
1) "stringa sg("acqua");" -> Richiama il costruttore di sg. 2) "return sg;" Costruttore di copia e distruttore di sg. Ma il costruttore di copia non e' stato definito, quindi viene richiamato quello di default, che inizializza il puntatore (char*) del nuovo oggetto in base al puntatore (char*) dell'oggetto sg... punteranno quindi entrambi alla stessa locazione di memoria, che pero' verra' deallocata subito dopo dal distruttore di sg. Ultima modifica di Bane : 05-12-2005 alle 15:50. |
|
|
|
|
|
#7 | ||
|
Member
Iscritto dal: Apr 2004
Messaggi: 130
|
Quote:
Dovrebbe esserci scritto qualcosa di simile, nel tuo testo, a meno che non sia vecchio/scadente. Quote:
Una possibile spiegazione potrebbe essere la sequenza: Codice:
>> Passaggio 1
---
stringa ff()
{
return stringa("acqua");
}
...
stringa sa = ff();
---
>> Passaggio 2
---
stringa sa = stringa("acqua");
---
>> Passaggio 3
---
stringa sa("acqua");
---
|
||
|
|
|
|
| Strumenti | |
|
|
Tutti gli orari sono GMT +1. Ora sono le: 12:45.



















