cpp:pointers
Differences
This shows you the differences between two versions of the page.
Both sides previous revisionPrevious revisionNext revision | Previous revision | ||
cpp:pointers [2021/04/27 03:54] – gthanos | cpp:pointers [2023/04/25 13:14] (current) – [Δείκτες null] gthanos | ||
---|---|---|---|
Line 62: | Line 62: | ||
===== Αριθμητική δεικτών ===== | ===== Αριθμητική δεικτών ===== | ||
- | Η αριθμητική δεικτών περιορίζεται μόνο στις πράξεις της πρόσθεσης και της αφαίρεσης. Μπορούμε να αυξήσουμε ή να μειώσουμε την τιμή ενός δείκτη κατά μία ή περισσότερες θέσεις. Η μεταβολή της τελικής τιμής του δείκτη | + | Η αριθμητική δεικτών περιορίζεται μόνο στις πράξεις της πρόσθεσης και της αφαίρεσης. Μπορούμε να αυξήσουμε ή να μειώσουμε την τιμή ενός δείκτη κατά μία ή περισσότερες θέσεις. Η μεταβολή της τελικής τιμής του δείκτη συνδέεται με τον μέγεθος του τύπου δεδομένων στον οποίο δείχνει ο δείκτης. Για παράδειγμα, |
<code c++> | <code c++> | ||
Line 71: | Line 71: | ||
</ | </ | ||
- | Αν και όλοι οι δείκτες καταλαμβάνουν τον ίδιο χώρο στη μνήμη (4-byte για συστήματα 32-bit και 8-byte για συστήματα 64-bit) διαφοροποιούνται ως προς το μέγεθος του περιεχόμενου των δεδομένων στα οποία δείχνουν. | + | Στο παραπάνω παράδειγμα η μεταβλητή //mychar// δείχνει σε δεδομένα μήκους ενός byte. Ας υποθέσουμε επίσης ότι η μεταβλητή //myshort// δείχνει σε δεδομένα μήκους 2 bytes (εξαρτάται από το hw) και η μεταβλητή //myint// δείχνει σε δεδομένα μήκους |
Ας επιχειρήσουμε να μεταβάλλουμε τις τιμές των παραπάνω δεικτών κατά μία θέση ως εξής: | Ας επιχειρήσουμε να μεταβάλλουμε τις τιμές των παραπάνω δεικτών κατά μία θέση ως εξής: | ||
Line 84: | Line 84: | ||
- η μεταβλητή myshort μετακινείται κατά // | - η μεταβλητή myshort μετακινείται κατά // | ||
- η μεταβλητή myint μετακινείται κατά // | - η μεταβλητή myint μετακινείται κατά // | ||
+ | |||
+ | Θα πρέπει να έχετε υπόψη ότι όλοι οι δείκτες καταλαμβάνουν τον ίδιο χώρο στη μνήμη (4-byte για συστήματα 32-bit και 8-byte για συστήματα 64-bit) ανεξάρτητα από τον τύπο δεδομένων στον οποίο δείχνουν. | ||
+ | |||
===== Δείκτες και πίνακες ===== | ===== Δείκτες και πίνακες ===== | ||
- | Η δήλωση ενός πίνακα ισοδυναμεί με ένα δείκτη στην πρώτη θέση του πίνακα. | + | Η δήλωση ενός πίνακα ισοδυναμεί με ένα δείκτη |
<code c> | <code c> | ||
Line 93: | Line 96: | ||
</ | </ | ||
- | Αν και ένας πίνακας μπορεί να αντιμετωπιστεί και ως δείκτης η βασική διαφορά πινάκων και δεικτών είναι ότι ο πίνακας δεν μπορεί να μεταβάλλει την διεύθυνση στην οποία δείχνει, | + | Αν και ένας πίνακας μπορεί να αντιμετωπιστεί και ως δείκτης η βασική διαφορά πινάκων και δεικτών είναι ότι ο πίνακας δεν μπορεί να μεταβάλλει την διεύθυνση στην οποία δείχνει, |
- | <code c> | + | |
+ | <code c> | ||
- | Δείτε το παρακάτω παράδειγμα όπου χρησιμοποιούνται πίνακας από ακεραίους και ένας δείκτης σε ακέραιο που δείχνει στις επιμέρους τιμές του πίνακα. | + | Δείτε το παρακάτω παράδειγμα όπου χρησιμοποιούνται πίνακας από ακεραίους και ένας δείκτης σε ακέραιο που δείχνει στις επιμέρους τιμές του πίνακα. |
<code cpp pointers-arrays.cpp> | <code cpp pointers-arrays.cpp> | ||
Line 106: | Line 110: | ||
int numbers[5]; | int numbers[5]; | ||
int * p; | int * p; | ||
- | p = numbers; | + | p = numbers; |
- | p++; *p = 20; // numbers[1] | + | p++; *p = 20; // numbers[1] |
- | p = & | + | p = & |
- | p = numbers + 3; *p = 40; // numbers[3] | + | p = numbers + 3; *p = 40; // numbers[3] |
- | p = numbers; | + | p = numbers; |
+ | | ||
for (int n=0; n<5; n++) | for (int n=0; n<5; n++) | ||
cout << numbers[n] << ", "; | cout << numbers[n] << ", "; | ||
Line 119: | Line 124: | ||
<WRAP tip 80% center round> | <WRAP tip 80% center round> | ||
- | Οι δείκτες οφείλουν να δείχνουν σε διευθύνσεις μνήμης που ανήκουν στην διεργασία που εκτελείται. Στην πραγματικότητα όμως οι δείκτες μπορούν να πάρουν οποιαδήποτε τιμή ακόμη και τιμές εκτός των ορίων της | + | Οι δείκτες οφείλουν να δείχνουν σε διευθύνσεις μνήμης που ανήκουν στη διεργασία που εκτελείται. Στην πραγματικότητα όμως οι δείκτες μπορούν να πάρουν οποιαδήποτε τιμή ακόμη και τιμές εκτός του εύρους διευθύνσεων που έχουν ανατεθεί από το λειτουργικό |
<code c++ intArrayPointer.cpp> | <code c++ intArrayPointer.cpp> | ||
#include < | #include < | ||
Line 128: | Line 134: | ||
*ptr = 10; | *ptr = 10; | ||
} | } | ||
- | </ | ||
- | </ | ||
- | |||
- | ===== Δείκτες αμετάβλητου περιεχομένου και δείκτες αμετάβλητης διεύθυνσης (const) ===== | ||
- | |||
- | Είδαμε ότι οι δείκτες είναι μεταβλητές που περιέχουν διευθύνσεις μνήμης στις οποίες αποθηκεύονται άλλες μεταβλητές. Μέσω των δεικτών μπορούμε να διαβάσουμε και να γράψουμε το περιεχόμενο μία διεύθυνσης μνήμης. Υπάρχουν όμως περιπτώσεις που ένας δείκτης είναι επιθυμητό να διαβάζει μόνο τα περιεχόμενα των διευθύνσεων μνήμης στα οποία δείχνει χωρίς να μπορεί να τα μεταβάλλει. Σε αυτή την περίπτωση αρκεί να δηλώσετε τον τύπο δεδομένων στον οποίο δείχνει ο δείκτης ως //const// ως εξής: | ||
- | |||
- | <code c> | ||
- | int x; | ||
- | int y = 10; | ||
- | const int * p = &y; | ||
- | x = *p; // μπορείτε να διαβάσετε το περιεχόμενο του δείκτη | ||
- | *p = x; // error: ΔΕΝ μπορείτε να μεταβάλλετε το περιεχόμενο του δείκτη. | ||
- | </ | ||
- | |||
- | Η χρήση δεικτών σε δεδομένα τύπου //const// είναι ευρέως διαδεδομένη σε συναρτήσεις, | ||
- | |||
- | <code c const_content.cpp> | ||
- | #include < | ||
- | using namespace std; | ||
- | |||
- | // stop pointer does not modify its contents | ||
- | void increment_all (int* start, const int* stop) { | ||
- | while (start != stop) { | ||
- | ++(*start); | ||
- | ++start; | ||
- | } | ||
- | } | ||
- | |||
- | // start & stop pointers do not modify their contents | ||
- | void print_all (const int* start, const int* stop) | ||
- | { | ||
- | while (start != stop) { | ||
- | cout << *start << endl; | ||
- | ++start; | ||
- | } | ||
- | } | ||
- | |||
- | int main () | ||
- | { | ||
- | int numbers[] = {10,20,30}; | ||
- | increment_all (numbers, | ||
- | print_all (numbers, | ||
- | return 0; | ||
- | } | ||
- | </ | ||
- | |||
- | Από τον παραπάνω παράδειγμα παρατηρήστε ότι ο δείκτης //start// στη συνάρτηση // | ||
- | |||
- | <code c> | ||
- | char c = ' | ||
- | char * const ptr = &c; | ||
- | </ | ||
- | |||
- | όπως στο παρακάτω παράδειγμα: | ||
- | |||
- | <code c const_address.cpp> | ||
- | #include < | ||
- | typedef struct { | ||
- | char name[32]; | ||
- | char age; | ||
- | } person_t; | ||
- | |||
- | void set_age(person_t * const p, int age) { | ||
- | p->age = age; // this is OK | ||
- | // | ||
- | } | ||
- | |||
- | int main() { | ||
- | person_t peter; | ||
- | strcpy(peter.name, | ||
- | set_age(& | ||
- | } | ||
- | </ | ||
- | |||
- | Τέλος εάν επιθυμείτε να έχετε ένα δείκτη ο οποίος δεν μεταβάλλει τη διεύθυνση στην οποία δείχνει και δεν μπορεί να μεταβάλλει το περιεχόμενο της διεύθυνσης αυτής τότε μπορείτε να τον δηλώσετε ως εξής: | ||
- | |||
- | <code c>char c = ' | ||
- | |||
- | Δείτε το παρακάτω παράδειγμα χρήσης ενός τέτοιου δείκτη | ||
- | |||
- | <code c const_address_const_ptr.cpp> | ||
- | #include < | ||
- | #include < | ||
- | using namespace std; | ||
- | |||
- | typedef struct { | ||
- | char name[32]; | ||
- | char age; | ||
- | } person_t; | ||
- | |||
- | void set_age(person_t * const p, int age) { | ||
- | p->age = age; // this is OK | ||
- | // | ||
- | } | ||
- | |||
- | int get_age(const person_t * const p) { | ||
- | // | ||
- | // | ||
- | return p-> | ||
- | } | ||
- | |||
- | int main() { | ||
- | person_t peter; | ||
- | strcpy(peter.name, | ||
- | set_age(& | ||
- | std::cout << "Age of" << peter.name << " is " << get_age(& | ||
- | } | ||
- | </ | ||
- | |||
- | |||
- | <WRAP center round tip 80%> | ||
- | Συμπερασματικά, | ||
- | <code c> | ||
- | int x; | ||
- | int * p1 = & | ||
- | const int * p2 = & | ||
- | int * const p3 = & | ||
- | const int * const p4 = & | ||
</ | </ | ||
</ | </ | ||
Line 282: | Line 169: | ||
{{ : | {{ : | ||
+ | |||
===== Δείκτες τύπου void ===== | ===== Δείκτες τύπου void ===== | ||
- | Ο δείκτης τύπου //void// είναι ένας ειδικός τύπος δείκτη ο οποίος έχει το πλεονέκτημα ότι μπορεί να μετατραπεί σε οποιονδήποτε τύπο δείκτη. Επίσης οποιοσδήποτε τύπος δείκτη μπορεί να μετατραπεί σε δείκτη τύπου //void//. Ο περιορισμός των δεικτών τύπου void είναι ότι δεν υπάρχει τρόπος να προσπελάσουμε το περιεχόμενο | + | Ο δείκτης τύπου //void// είναι ένας ειδικός τύπος δείκτη ο οποίος έχει το πλεονέκτημα ότι μπορεί να μετατραπεί σε οποιονδήποτε τύπο δείκτη. Επίσης οποιοσδήποτε τύπος δείκτη μπορεί να μετατραπεί σε δείκτη τύπου //void//. Ο περιορισμός των δεικτών τύπου void είναι ότι δεν υπάρχει τρόπος να προσπελάσουμε |
<code c void_ptr.cpp> | <code c void_ptr.cpp> | ||
Line 312: | Line 200: | ||
<WRAP center round tip 80%> | <WRAP center round tip 80%> | ||
- | Δεν | + | Δεν |
</ | </ | ||
- | ===== null pointers | + | ===== Δείκτες με τιμή NULL ===== |
Δείκτες που δεν είναι αρχικοποιημένοι συχνά αρχικοποιούνται στην τιμή 0 ή NULL. Η C++ εκτός από τις τιμές 0 και null παρέχει και την δεσμευμένη λέξη **nullptr** που αντιπροσωπεύει και αυτή την τιμή 0 για ένα δείκτη. Οι παρακάτω ορισμοί είναι ισοδύναμοι. | Δείκτες που δεν είναι αρχικοποιημένοι συχνά αρχικοποιούνται στην τιμή 0 ή NULL. Η C++ εκτός από τις τιμές 0 και null παρέχει και την δεσμευμένη λέξη **nullptr** που αντιπροσωπεύει και αυτή την τιμή 0 για ένα δείκτη. Οι παρακάτω ορισμοί είναι ισοδύναμοι. | ||
Line 321: | Line 209: | ||
<code c++ nullptr.cpp> | <code c++ nullptr.cpp> | ||
#include < | #include < | ||
+ | |||
char *p = 0; | char *p = 0; | ||
char *q = NULL; | char *q = NULL; |
cpp/pointers.1619495695.txt.gz · Last modified: 2021/04/27 02:54 (external edit)