cpp:copy_constructors
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
cpp:copy_constructors [2020/04/14 15:24] – gthanos | cpp:copy_constructors [2022/05/12 19:41] (current) – [Πιο σύνθετες περιπτώσεις] gthanos | ||
---|---|---|---|
Line 23: | Line 23: | ||
int getHeight(); | int getHeight(); | ||
}; | }; | ||
+ | </ | ||
+ | |||
+ | <code cpp Rectangle.cpp> | ||
+ | #include " | ||
Rectangle:: | Rectangle:: | ||
Line 29: | Line 33: | ||
} | } | ||
- | Rectangle:: | + | Rectangle:: |
cout << " | cout << " | ||
width = s; height = s; | width = s; height = s; | ||
Line 64: | Line 68: | ||
{{ : | {{ : | ||
- | Το ερώτημα είναι με ποιό τρόπο γίνεται η δημιουργία του αντιγράφου του αρχικού αντικειμένου στο //stack//. Μπορείτε να δηλώσετε τον δικό σας κατασκευαστή αντιγραφέα (λεπτομέρειες πιο κάτω..) ή να αφήσετε τον // | + | Το ερώτημα είναι με ποιό τρόπο γίνεται η δημιουργία του αντιγράφου του αρχικού αντικειμένου στο //stack//. Μπορείτε να δηλώσετε τον δικό σας κατασκευαστή αντιγραφέα (λεπτομέρειες πιο κάτω..) ή να αφήσετε τον // |
===== Ορισμός ενός κατασκευαστή αντιγραφέα ===== | ===== Ορισμός ενός κατασκευαστή αντιγραφέα ===== | ||
Line 76: | Line 80: | ||
} | } | ||
</ | </ | ||
- | ή ο παρακάτω | + | ή ισοδύναμα |
<code cpp> | <code cpp> | ||
Rectangle:: | Rectangle:: | ||
Line 84: | Line 88: | ||
</ | </ | ||
- | Και οι δύο παραπάνω κατασκευαστές δηλώνουν ένα κατασκευαστή αντιγραφέα. | + | Και οι δύο παραπάνω κατασκευαστές |
<code cpp Rectangle.hpp> | <code cpp Rectangle.hpp> | ||
Line 99: | Line 103: | ||
Rectangle(int s); | Rectangle(int s); | ||
Rectangle(); | Rectangle(); | ||
- | Rectangle(Rectangle& | + | Rectangle(const Rectangle& |
void setWidth(int w); | void setWidth(int w); | ||
void setHeight(int h); | void setHeight(int h); | ||
Line 105: | Line 109: | ||
int getHeight(); | int getHeight(); | ||
}; | }; | ||
+ | </ | ||
+ | |||
+ | <code cpp Rectangle.cpp> | ||
+ | #include " | ||
Rectangle:: | Rectangle:: | ||
Line 111: | Line 119: | ||
} | } | ||
- | Rectangle:: | + | Rectangle:: |
cout << " | cout << " | ||
width = s; height = s; | width = s; height = s; | ||
Line 121: | Line 129: | ||
} | } | ||
- | Rectangle:: | + | Rectangle:: |
- | cout << Calling copy constructor" | + | cout << |
width = r.width; | width = r.width; | ||
height = r.height; | height = r.height; | ||
Line 137: | Line 145: | ||
</ | </ | ||
- | ==== Άλλη περίπτωση κλήση | + | ==== Άλλη περίπτωση κλήσης κατασκευαστή αντιγραφέα |
- | Μία άλλη περίπτωση κατά την οποία θα κληθεί o κατασκευαστής αντιγραφέας είναι η παρακάτω. Εδώ η δήλωση της μεταβλητής r2 συμπίπτει με την αρχικοποίηση του αντικειμένου. Σε αυτή την περίπτωση καλείται ο κατασκευαστής αντιγραφέας με όρισμα το //r1//. | + | Μία άλλη περίπτωση κατά την οποία θα κληθεί o κατασκευαστής αντιγραφέας είναι η παρακάτω. |
<code cpp CopyRectangle.cpp> | <code cpp CopyRectangle.cpp> | ||
Line 149: | Line 157: | ||
} | } | ||
</ | </ | ||
+ | |||
+ | Εδώ η δήλωση της μεταβλητής r2 συμπίπτει με την αρχικοποίηση του αντικειμένου. Σε αυτή την περίπτωση καλείται ο κατασκευαστής αντιγραφέας με όρισμα το // | ||
<WRAP center round tip 80%> | <WRAP center round tip 80%> | ||
- | Το παραπάνω είναι λειτουργικά ισοδύναμο με το παρακάτω. | + | Το παραπάνω είναι λειτουργικά ισοδύναμο με το εξής: |
<code cpp CopyRectangle.cpp> | <code cpp CopyRectangle.cpp> | ||
#include " | #include " | ||
Line 161: | Line 171: | ||
} | } | ||
</ | </ | ||
- | Για τον μεταγλωττιστή όμως οι κώδικες είναι διαφορετικοί. Στην πρώτη περίπτωση καλείται ο κατασκευαστής αντιγραφέας (//copy constructor// | + | Για τον μεταγλωττιστή όμως οι δύο |
+ | |||
+ | **Σημείωση: | ||
</ | </ | ||
- | ===== Μια πιο σύνθετη περίπτωση | + | ===== Πιο σύνθετες περιπτώσεις ===== |
+ | |||
+ | ==== 1η Περίπτωση | ||
- | Στις περιπτώσεις που υπάρχουν πεδία δείκτες | + | Στις περιπτώσεις που υπάρχουν πεδία |
Στο παρακάτω παράδειγμα ορίζουμε την κλάση //Point// η οποία αντιπροσωπεύει ένα σημείο στο δισδιάστατο χώρο. | Στο παρακάτω παράδειγμα ορίζουμε την κλάση //Point// η οποία αντιπροσωπεύει ένα σημείο στο δισδιάστατο χώρο. | ||
Line 196: | Line 210: | ||
void setX(int vx) { x = vx; } | void setX(int vx) { x = vx; } | ||
void setY(int vy) { y = vy; } | void setY(int vy) { y = vy; } | ||
- | int getX() { return x; } | + | int getX() |
- | int getY() { return y; } | + | int getY() |
}; | }; | ||
</ | </ | ||
Line 220: | Line 234: | ||
Rectangle(); | Rectangle(); | ||
~Rectangle(); | ~Rectangle(); | ||
- | Rectangle(Rectangle &r); | ||
void setWidth(int w); | void setWidth(int w); | ||
void setHeight(int h); | void setHeight(int h); | ||
Line 226: | Line 239: | ||
int getHeight(); | int getHeight(); | ||
void setOrigin(Point *p); | void setOrigin(Point *p); | ||
- | Point *getOrigin(); | + | Point *getOrigin() |
}; | }; | ||
+ | </ | ||
+ | |||
+ | <code cpp Rectangle.cpp> | ||
+ | #include " | ||
Rectangle:: | Rectangle:: | ||
Line 256: | Line 273: | ||
int Rectangle:: | int Rectangle:: | ||
void Rectangle:: | void Rectangle:: | ||
- | Point *Rectangle:: | + | Point *Rectangle:: |
</ | </ | ||
- | <code cpp MoveOrigin.cpp> | + | <code cpp CopyRectangle.cpp> |
#include " | #include " | ||
- | |||
- | int moveOrigin(Rectangle &r, int dx, int dy) { | ||
- | Point *p = r.getOrigin(); | ||
- | p-> | ||
- | p-> | ||
- | } | ||
int main() { | int main() { | ||
- | Point p{10,5}; | + | Point p{5,5}; |
Rectangle r1{5, | Rectangle r1{5, | ||
Rectangle r2 = r1; | Rectangle r2 = r1; | ||
- | | ||
- | moveOrigin(r1, | ||
} | } | ||
</ | </ | ||
- | Ο παραπάνω κώδικας αντιγράφει στο αντικείμενο //r2// τα πεδία του αντικειμένου //r1// πεδίο προς πεδίο. Αυτό σημαίνει ότι τα αντικείμενα //r1// και //r2// μοιράζονται το ίδιο αντικείμενο τύπου //Point//. Ισχύουν επομένως τα εξής: | + | Ο παραπάνω κώδικας |
* Εάν μεταβληθούν οι συντεταγμένες του //Point// από το αντικείμενο //r1//, η μεταβολή θα ισχύει και για το αντικείμενο // | * Εάν μεταβληθούν οι συντεταγμένες του //Point// από το αντικείμενο //r1//, η μεταβολή θα ισχύει και για το αντικείμενο // | ||
- | * Κατά την έξοδο από τη συνάρτηση main, το αντικείμενο //r1// θα καταστραφεί ελευθερώνοντας τη δεσμευμένη μνήμη για το πεδίο του //origin//. Η προσπάθεια καταστρροφής του αντικειμένου //r2// θα οδηγήσει σε σφάλμα διότι θα προσπαθήσει να ελευθερώσει μία περιοχή μνήμης που έχει ήδη ελευθερωθεί κατά την καταστροφή του //r1//. Το σφάλμα που εκτυπώνεται όταν το πρόγραμμα εκτελεστεί είναι το εξής: | + | * Κατά την έξοδο από τη συνάρτηση main, το αντικείμενο //r1// θα καταστραφεί ελευθερώνοντας τη δεσμευμένη μνήμη για το πεδίο του //origin//. Η προσπάθεια καταστροφής του αντικειμένου //r2// θα οδηγήσει σε σφάλμα διότι θα προσπαθήσει να ελευθερώσει μία περιοχή μνήμης που έχει ήδη ελευθερωθεί κατά την καταστροφή του //r1//. Το σφάλμα που εκτυπώνεται όταν το πρόγραμμα εκτελεστεί είναι το εξής: |
< | < | ||
- | *** Error in `./ | + | free(): |
- | Ακυρώθηκε | + | Aborted |
</ | </ | ||
Line 292: | Line 301: | ||
Ο προτεινόμενος κατασκευαστής αντιγραφέας δίνεται παρακάτω: | Ο προτεινόμενος κατασκευαστής αντιγραφέας δίνεται παρακάτω: | ||
<code cpp> | <code cpp> | ||
- | Rectangle:: | + | Rectangle:: |
width = r.width; | width = r.width; | ||
height = r.height; | height = r.height; | ||
Line 307: | Line 316: | ||
</ | </ | ||
+ | ==== 2η Περίπτωση - Κλήση συνάρτησης με παράμετρο αντικείμενο που περιέχει δυναμικά δεσμευμένα πεδία ==== | ||
+ | <code cpp MoveOrigin.cpp> | ||
+ | #include " | ||
+ | int moveOrigin(Rectangle r, int dx, int dy) { | ||
+ | Point *p = r.getOrigin(); | ||
+ | p-> | ||
+ | p-> | ||
+ | } | ||
+ | int main() { | ||
+ | Point p{5,5}; | ||
+ | Rectangle r1{5, | ||
+ | | ||
+ | moveOrigin(r1, | ||
+ | } | ||
+ | </ | ||
+ | Η κλήση της συνάρτησης moveOrigin θα απαιτήσει τη δημιουργία ενός αντιγράφου του r1 στο stack για τις ανάγκες της εκτέλεσης της συγκεκριμένης συνάρτησης. Εφόσον δεν έχουμε δημιουργήσει κατασκευαστή αντιγραφέα, | ||
+ | Κατά την έξοδο από τη συνάρτηση η κοινή αυτή περιοχή θα καταστραφεί, | ||
+ | < | ||
+ | free(): double free detected in tcache 2 | ||
+ | Aborted (core dumped) | ||
+ | </ | ||
+ | Η δημιουργία του κατασκευαστή αντιγραφέα που προτάθηκε παραπάνω λύνει και αυτό το πρόβλημα. |
cpp/copy_constructors.1586877879.txt.gz · Last modified: 2020/04/14 14:24 (external edit)