cvb~ Δημοσ. 4 Απριλίου 2024 Δημοσ. 4 Απριλίου 2024 Καλησπέρα! Έχουμε τον παρακάτω κώδικα που μας ζητάει 3 αριθμούς και μετά υπολογίζει και εμφανίζει το άθροισμα τους. H vector_init(Vector& v, int s) όπως μπορείτε να δείτε χρησιμοποιεί το new για να κατανείμει χώρο μνήμης για ένα array με s σε πλήθος doubles που θα αποθηκευτούν οι αριθμοί. Πως και που πρέπει να αποδεσμευτεί αυτός ο χώρος μνήμης; Ευχαριστώ! // tour22.cpp #include <iostream> using namespace std; struct Vector { double* elem; // pointer to elements int sz; // number of elements }; void vector_init(Vector& v, int s) // initialize a Vector { v.elem = new double[s]; // allocate an array of s doubles v.sz = s; } double read_and_sum(int s) // read s integers from cin and return their sum; s is assumed to be positive { Vector v; vector_init(v, s); // allocate s elements for v for (int i = 0; i != s; ++i) cin >> v.elem[i]; // read into elements double sum = 0; for (int i = 0; i != s; ++i) sum += v.elem[i]; // compute the sum of the elements return sum; } int main(void) { double sum = 0; cout << "Enter 3 numbers:\n"; sum = read_and_sum(3); cout << "Sum: " << sum << '\n'; return 0; }
Moderators Λύση Kercyn Δημοσ. 5 Απριλίου 2024 Moderators Λύση Δημοσ. 5 Απριλίου 2024 Δεν ξέρω πού θα μπορούσες να το βάλεις στο συγκεκριμένο κώδικα ούτως ώστε: να διασφαλιστείς από πιθανά προβλήματα να δουλεύει η εφαρμογή σου να βγάζει νόημα Ενδεχομένως θα μπορούσες να το βάλεις στο read_and_sum, μιας και το function έχει ήδη side effects, οπότε θα μπορούσες να το "σώσεις" όπως μπορείς. Βέβαια, σε περίπτωση που έχεις exception πριν φτάσεις στο delete, τότε πιθανότατα θα έχεις leak. Από κει και πέρα, αυτό που θα μπορούσες να κάνεις είναι να σπάσεις το read_and_sum σε 3 κομμάτια: το ένα κομμάτι το έχεις ήδη, είναι η αρχικοποίηση του πίνακα. Το δεύτερο είναι το διάβασμα του input από το χρήστη, και το 3ο είναι ο υπολογισμός του αθροίσματος. Έτσι, θα μπορούσες να έχεις τα παρακάτω στη main σου: αρχικοποίηση πίνακα διάβασμα αριθμών υπολογισμός αθροίσματος αποδέσμευση της μνήμης Τώρα, αν θες να το πας ένα βήμα παραπέρα, μπορείς να δεις και τα εξής: Χρήση smart pointers. Ο μόνος λόγος (ίσως) να χρησιμοποιείς raw pointers είναι μόνο αν είναι non-owning. Οι owning raw pointers είναι ιδιαίτερα επικίνδυνοι (μπορεί να ξεχάσεις να τους ελευθερώσεις, προβλήματα null pointer dereferencing, exception safety κλπ κλπ). Να χρησιμοποιήσεις το std::vector (αν δεν ξέρεις το μέγεθος της λίστας σου) ή το std::array αν το ξέρεις. Αν θες σώνει και ντε το δικό σου class/struct, να βάλεις τη λειτουργικότητα του vector_init στον constructor του Vector (το οποίο ενδεχομένως και να μη χρειάζεται καν, αναλόγως της υλοποίησής σου) Όλα τα παραπάνω με επιφύλαξη, καθώς έχω να χρησιμοποιήσω C++ αρκετά χρόνια οπότε μπορεί κάτι να μου έχει ξεφύγει ή να μην το θυμάμαι σωστά. 1
n3a Δημοσ. 5 Απριλίου 2024 Δημοσ. 5 Απριλίου 2024 Στο function read_and_sum(), δημιουργείς έναν πίνακα με doubles και αποθηκεύεις την διεύθυνσή του στο "v.elem". Πριν τελειώσει το function μπορείς να ελευθερώσεις τον πίνακα από το heap ως εξής: double read_and_sum(int s) // read s integers from cin and return their sum; s is assumed to be positive { Vector v; vector_init(v, s); // allocate s elements for v for (int i = 0; i != s; ++i) cin >> v.elem[i]; // read into elements double sum = 0; for (int i = 0; i != s; ++i) sum += v.elem[i]; // compute the sum of the elements delete[] v.elem; return sum; } 1
cvb~ Δημοσ. 5 Απριλίου 2024 Μέλος Δημοσ. 5 Απριλίου 2024 (επεξεργασμένο) Έχεις δίκιο, αυτό ζητούσα. Ο χαζός δοκίμαζα σκέτο "delete[] v; και δεν μου το δεχόταν ο μεταγλωττιστής. Σε ευχαριστώ! Επεξ/σία 5 Απριλίου 2024 από cvb~
παπι Δημοσ. 6 Απριλίου 2024 Δημοσ. 6 Απριλίου 2024 Απο την ωρα που εχεις vector_init το οποιο κανει την δουλεια του allocator, δε μπορεις να βαλεις delete[]. Πρεπει να φτιαξεις δικο σου deallocator, τυπου vector_destroy/vector_release/vector_delete whateva. 1
Προτεινόμενες αναρτήσεις
Δημιουργήστε ένα λογαριασμό ή συνδεθείτε για να σχολιάσετε
Πρέπει να είστε μέλος για να αφήσετε σχόλιο
Δημιουργία λογαριασμού
Εγγραφείτε με νέο λογαριασμό στην κοινότητα μας. Είναι πανεύκολο!
Δημιουργία νέου λογαριασμούΣύνδεση
Έχετε ήδη λογαριασμό; Συνδεθείτε εδώ.
Συνδεθείτε τώρα