Torna indietro   Hardware Upgrade Forum > Software > Programmazione

Dreame Aqua10 Ultra Roller, la pulizia di casa con un rullo
Dreame Aqua10 Ultra Roller, la pulizia di casa con un rullo
Il più recente robot per la pulizia domestica di Dreame, modello Aqua10 Ultra Roller, abbina un potente motore di aspirazione della polvere a un sofisticato sistema di lavaggio con rullo integrato. Il tutto governato dalla logica di intelligenza artificiale, per i migliori risultati
Recensione Realme 15 Pro Game Of Thrones: un vero cimelio tech per pochi eletti
Recensione Realme 15 Pro Game Of Thrones: un vero cimelio tech per pochi eletti
Siamo volati fino a Belfast, capitale dell'Irlanda Del Nord, per scoprire il nuovo Realme 15 Pro 5G Game Of Thrones Limited Edition. Una partnership coi fiocchi, quella tra Realme e HBO, un esercizio di stile davvero ben riuscito. Ma vi raccontiamo tutto nel nostro articolo
GIGABYTE GAMING A16, Raptor Lake e RTX 5060 Laptop insieme per giocare al giusto prezzo
GIGABYTE GAMING A16, Raptor Lake e RTX 5060 Laptop insieme per giocare al giusto prezzo
Il Gigabyte Gaming A16 offre un buon equilibrio tra prestazioni e prezzo: con Core i7-13620H e RTX 5060 Laptop garantisce gaming fluido in Full HD/1440p e supporto DLSS 4. Display 165 Hz reattivo, buona autonomia e raffreddamento efficace; peccano però le USB e la qualità cromatica del pannello. Prezzo: circa 1200€.
Tutti gli articoli Tutte le news

Vai al Forum
Rispondi
 
Strumenti
Old 03-09-2010, 16:39   #1
Zero-Giulio
Member
 
Iscritto dal: May 2007
Messaggi: 292
[C++] Conflitti tra costruttore di copia e operatore di assegnamento

Per esporre il mio problema, incollo subito una classe semplice (ridotta al midollo):

Codice:
class obj {

    private:

        int arg1;

    public:

        obj ()
            : arg1 (1) {}

        obj (const string & s)
            : arg1 (15) {}

        const obj & operator = (const obj & p) {
            arg1 = p.arg1;
            return *this;
        }

        const obj & operator = (const string & s) {
            arg1 = 15;
            return *this;
        }

        int get () {
            return arg1;
        }

        void set (int arg) {
            arg1 = arg;
        }

};
La mia domanda è molto semplice:

perchè

Codice:
obj p;
p = "ciao";
funziona, e

Codice:
obj p ("ciao");
funziona pure, mentre

Codice:
obj p = "ciao";
non funziona?

Ultima modifica di Zero-Giulio : 03-09-2010 alle 16:46.
Zero-Giulio è offline   Rispondi citando il messaggio o parte di esso
Old 03-09-2010, 18:34   #2
DanieleC88
Senior Member
 
L'Avatar di DanieleC88
 
Iscritto dal: Jun 2002
Città: Dublin
Messaggi: 5989
Perché la tua prima versione chiama il costruttore obj::obj() e poi, una volta costruito l'oggetto, chiama il metodo obj::operator=(const std::string&), costruendo dinamicamente un oggetto std::string a partire dal const char[] che provi ad assegnarvi. (Esiste un costruttore di std::string che accetta un const char*, per cui tutto ciò è fattibile.)
La seconda versione invece cerca di costruire l'oggetto dando come argomento del costruttore un const char[]: per gli stessi motivi di prima, ciò riesce con il costruttore obj::obj(const std::string&).
La terza invece cerca di utilizzare un costruttore di copia. Siccome un const char* non è un'istanza di obj valida, il compilatore dovrebbe tentare di costruire prima un'istanza di obj passando come argomento al costruttore un const char*, il che però non è possibile visto che non esiste un costruttore che lo accetti come parametro.

Il punto è che ci vorrebbero due passaggi anziché uno per invocare il costruttore:
  1. parti da un const char*
  2. costruisci un'istanza di std::string usando std::string::string(const char*)
  3. costruisci un'istanza di obj usando obj::obj(std::string&), con riferimento all'istanza di std::string costruita al passo precedente
  4. invochi il costruttore di copia di obj

Il passaggio critico è dato dal fatto che il compilatore cercherà implicitamente di tradurre la tua dichiarazione in:
Codice:
obj p = obj("ciao");
e non in:
Codice:
obj p = obj(std::string("ciao"));
come invece c'è bisogno di fare. Il compilatore non ha diritto di prendere questa decisione per te, e mancando un costruttore adatto ti sputa fuori l'errore di compilazione.

Diciamo che puoi risolvere in due modi: o aggiungi un costruttore di obj che accetti un const char*, facendo così scomparire l'errore alla radice, oppure costruisci esplicitamente un'istanza di std::string da passare implicitamente al costruttore:
Codice:
obj p = std::string("ciao");
ciao
__________________

C'ho certi cazzi Mafa' che manco tu che sei pratica li hai visti mai!

Ultima modifica di DanieleC88 : 03-09-2010 alle 18:36.
DanieleC88 è offline   Rispondi citando il messaggio o parte di esso
Old 04-09-2010, 14:29   #3
Zero-Giulio
Member
 
Iscritto dal: May 2007
Messaggi: 292
Continuo ad avere dubbi.
Sapevo che

Codice:
obj p = "ciao";
veniva tradotto in

Codice:
obj p = obj ("ciao");
ma non capisco poi quale sia il problema.
Infatti

Codice:
obj ("ciao")
è una cosa che il compilatore riesce a fare (capisce che deve trasformare il char * in string), infatti
Codice:
obj p ("ciao")
funziona, e anche l'operatore di assegnamento funziona.
Non capisco qual'è la differenza, perchè qui

Codice:
obj p ("ciao")
il compilatore capisce che deve trasformare un char * in una string mentre in

Codice:
obj p = obj ("ciao")
non lo capisce?
Perchè il copy constructor una volta è intelligente e una volta no?
Zero-Giulio è offline   Rispondi citando il messaggio o parte di esso
Old 04-09-2010, 19:51   #4
DanieleC88
Senior Member
 
L'Avatar di DanieleC88
 
Iscritto dal: Jun 2002
Città: Dublin
Messaggi: 5989
No, vedi che ho già risposto a quello che mi chiedi...

Se tu dici:
Codice:
obj p("ciao");
allora il compilatore cerca il costruttore più adatto. Siccome non c'è un costruttore che accetti un const char[], ripiega su quello che accetta un std::string. Questo lo fa perché esiste un costruttore di std::string che accetti un const char*, e quindi non ha di che lamentarsi. Il compilatore, dunque, pensa:
Codice:
obj p(std::string("ciao"));
In questo caso, il passaggio implicito è la costruzione di un oggetto di tipo std::string, perché tutto il resto è già fornito al compilatore.

Quando, invece, tu dici:
Codice:
obj p = "ciao";
il compilatore pensa:
Codice:
obj p = obj("ciao");
e basta: quindi, nessun adattamento di "ciao", che è un const char[], ad un std::string. In questo caso, il passaggio implicito è la costruzione di un oggetto di tipo obj, ma non è sufficiente: ci vorrebbe infatti un secondo passaggio implicito, che è quello della costruzione di un oggetto std::string a partire dal const char[].

Nel caso precedente il compilatore doveva fare una sola decisione (e la fa), in quest'ultimo caso invece ne dovrebbe fare due, una conseguenza dell'altra.
Se tu potessi fare più di una "scelta" di questo tipo, allora ne dovresti poter fare a catena: potevi ad esempio scrivere una cosa del genere:
Codice:
obj p = 123;
ed aspettarti che venga "trasformata" dal compilatore in:
Codice:
obj p = "123";
e quindi, ancora, in:
Codice:
obj p = std::string("123");
O no?
__________________

C'ho certi cazzi Mafa' che manco tu che sei pratica li hai visti mai!
DanieleC88 è offline   Rispondi citando il messaggio o parte di esso
 Rispondi


Dreame Aqua10 Ultra Roller, la pulizia di casa con un rullo Dreame Aqua10 Ultra Roller, la pulizia di casa c...
Recensione Realme 15 Pro Game Of Thrones: un vero cimelio tech per pochi eletti Recensione Realme 15 Pro Game Of Thrones: un ver...
GIGABYTE GAMING A16, Raptor Lake e RTX 5060 Laptop insieme per giocare al giusto prezzo GIGABYTE GAMING A16, Raptor Lake e RTX 5060 Lapt...
iPhone 17 Pro: più di uno smartphone. È uno studio di produzione in formato tascabile iPhone 17 Pro: più di uno smartphone. &Eg...
Intel Panther Lake: i processori per i notebook del 2026 Intel Panther Lake: i processori per i notebook ...
Sempre più vicino il lancio (e il...
L'incredibile ottimizzazione di Battlefi...
La NASA aprirà il contratto per i...
PS5 Pro CFI-7121: ecco tutte le differen...
SpaceX ha raggiunto il traguardo dei 10....
ROG Xbox Ally e Ally X sono care, ma il ...
Intesa Sanpaolo: stop all'app Mobile su ...
Gli USA costruiscono chip AI… ma non pos...
Grazie a Ericsson, l'italia è in ...
CMF by Nothing Buds 2a: ANC da 42 dB e o...
Il futuro di Porsche è ibrido: il...
Apple lancia i nuovi MacBook Pro e iPad ...
Free software e biologia: intervista al ...
Dongfeng Box bocciata da Euro NCAP: solo...
Parigi abbandona ufficialmente Eurodrone...
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: 03:50.


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