1 / 21

ΜΑΘ 106/3122 Γλώσσα Προγραμματισμού

ΜΑΘ 106/3122 Γλώσσα Προγραμματισμού. Δείκτες ( Pointers). Δείκτες. Τι είναι : τύπος μεταβλητών (όπως integer) Τι αποθηκεύουν : την διεύθυνση στη μνήμη άλλων μεταβλητών Τι χρειάζονται : Κυρίως, για δυναμική διαχείριση μνήμης και δυναμικές δομές δεδομένων Call-by-reference

gage-bauer
Download Presentation

ΜΑΘ 106/3122 Γλώσσα Προγραμματισμού

An Image/Link below is provided (as is) to download presentation Download Policy: Content on the Website is provided to you AS IS for your information and personal use and may not be sold / licensed / shared on other websites without getting consent from its author. Content is provided to you AS IS for your information and personal use only. Download presentation by click this link. While downloading, if for some reason you are not able to download a presentation, the publisher may have deleted the file from their server. During download, if you can't get a presentation, the file might be deleted by the publisher.

E N D

Presentation Transcript


  1. ΜΑΘ 106/3122Γλώσσα Προγραμματισμού Δείκτες (Pointers)

  2. Δείκτες Τι είναι: τύπος μεταβλητών (όπως integer) Τι αποθηκεύουν: την διεύθυνση στη μνήμη άλλων μεταβλητών Τι χρειάζονται: Κυρίως, για δυναμική διαχείριση μνήμης και δυναμικές δομές δεδομένων Call-by-reference Στενή σχέση με τους πίνακες Πηγή πολλών λαθών που δύσκολα ανιχνεύονται

  3. Δείκτες count countPtr count 7 7 int Counter = 7; int *PointerToCounter; PointerToCounter = &Counter;

  4. Δήλωση Δεικτών: • Το *χρησιμοποιείται για να δηλώσει δείκτη int *myPtr; • Ορίζει δείκτη σεint (pointer of type int *) • Δήλωση πολλών μεταβλητών τύπου δείκτη απαιτεί τη χρήση ενός *μπροστά από κάθε μία: int *myPtr1, *myPtr2; • Μπορούμε να δηλώσουμε δείκτη σε οποιοδήποτε είδος μεταβλητής (ακόμα και άλλο δείκτη) double *myPtr2; char **myPtr2;

  5. Τελεστές Δεικτών • &μεταβλητή • q = &a; • Επέστρεψε την διεύθυνση στην οποία είναι αποθηκευμένη η μεταβλητή • *δείκτης • a = *q; • Επέστρεψε τα περιεχόμενα της διεύθυνσης στην οποία δείχνει ο δείκτης

  6. Τελεστές Δεικτών yptr y y 5 yPtr Η διεύθυνση του yείναι η τιμή της yptr 500000 600000 600000 5 • & (address operator) int y = 5; int *yPtr; yPtr = &y; // O yPtr παίρνει την διεύθυνση y

  7. Τελεστές Δεικτών • * (indirection/dereferencing operator) int y = 5; int *yPtr; yPtr = &y; *yPtr = *yPtr * 2; • Μπορώ να αλλάζω την τιμή των μεταβλητών με τη βοήθεια δεικτών • Η τιμή του y γίνεται ίση με 10 • Η μεταβλητή y αλλάζει τιμή, χωρίς στον κώδικα να φαίνεται αυτό άμεσα σε μια εντολή που εμφανίζει την y! • H y δηλαδή, αποκτά ένα «ψευδώνυμο»: το *yPtr • *και&είναι ανάποδα και αλληλοακυρώνονται

  8. Ανάθεση δεικτών • Ένας δείκτης μπορεί να ανατεθεί σε άλλο δείκτη ίδιου τύπου: int a = 4; int * p; p = &a; int * q = p; Η εντολή int * q = p; ορίζει το q ως δείκτη σε int και του δίνει αρχική τιμή το p. Τα q και p δείχνουν την ίδια θέση μνήμης και, άρα, *q και *p δείχνουν στην ίδια μεταβλητή (η a).

  9. Αρχικοποίηση δεικτών • Ένας δείκτης που δεν του αποδοθεί αρχική τιμή δείχνει σε τυχαία περιοχή μνήμης. Η προσπέλαση στην τυχαία θέση μνήμης αυτή θα προκαλέσει σφάλμα εκτέλεσης, αν η συγκεκριμένη θέση δεν έχει δοθεί από το λειτουργικό σύστημα στο πρόγραμμά. Αν έχει δοθεί, και γράψουμε στη θέση μνήμης αυτή, θα “πανωγράψουμε” άλλη “δική μας” μεταβλητή χωρίς σφάλμα εκτέλεσης. • Σε ένα οποιοδήποτε δείκτη μπορεί να γίνει απόδοση της τιμής 0 . Το 0 δεν αποτελεί αριθμό θέσης μνήμης και, επομένως, η ανάθεση αυτή υποδηλώνει ότι ο δείκτης δε δείχνει σε κανένα αντικείμενο. Σε τέτοιο δείκτη (null pointer) η δράση του (*) δεν επιτρέπεται (προκαλεί λάθος κατά την εκτέλεση αλλά όχι κατά τη μεταγλώττιση του προγράμματος). • Η σύγκριση ενός άγνωστου δείκτη (π.χ. όρισμα συνάρτησης) με το 0 πρέπει να προηγείται οποιασδήποτε απόπειρας προσπέλασης της θέσης μνήμης στην οποία θεωρούμε ότι δείχνει. Προσέξτε ότι σε αυτό το σημείο υπάρχει μία βασική διαφορά με την αναφορά: ένας δείκτης μπορεί να μη συνδέεται με κανένα αντικείμενο ενώ, αντίθετα, μία αναφορά είναι οπωσδήποτε συνδεδεμένη με κάποια ποσότητα.

  10. 1 /* The address of a is the value of aPtr. 2 Using the & and * operators */ 3 #include <stdio.h> 4 The * operator returns an alias to what its operand points to. aPtr points to a, so *aPtr returns a. 5 int main() 6 { Notice how * and & are inverses 7 int a; /* a is an integer */ 8 int *aPtr; /* aPtr is a pointer to an integer */ 9 10 a = 7; 11 aPtr = &a; /* aPtr set to address of a */ 12 13 printf( "The address of a is %p" 14 "\nThe value of aPtr is %p", &a, aPtr ); 15 16 printf( "\n\nThe value of a is %d" 17 "\nThe value of *aPtr is %d", a, *aPtr ); 18 19 printf( "\n\nShowing that * and & are inverses of " 20 "each other.\n&*aPtr = %p" 21 "\n*&aPtr = %p\n", &*aPtr, *&aPtr ); 22 23 return 0; 24 } The address of a is 0012FF88 The value of aPtr is 0012FF88 The value of a is 7 The value of *aPtr is 7 Proving that * and & are complements of each other. &*aPtr = 0012FF88 *&aPtr = 0012FF88

  11. 1 /* 2 Cube a variable using call-by-reference Notice how the address of number is given - cubeByReference expects a pointer (an address of a variable). 3 with a pointer argument */ 4 5 #include <stdio.h> 6 Inside cubeByReference, *nPtr is used (*nPtr is number). 7 void cubeByReference( int * ); /* prototype */ 8 9 int main() 10 { 11 int number = 5; 12 13 printf( "The original value of number is %d", number ); 14 cubeByReference( &number ); 15 printf( "\nThe new value of number is %d\n", number ); 16 17 return 0; 18 } 19 20 void cubeByReference( int *nPtr ) 21 { 22 *nPtr = *nPtr * *nPtr * *nPtr; /* cube number in main */ 23 } Notice that the function prototype takes a pointer to an integer (int *). The original value of number is 5 The new value of number is 125

  12. Sort με χρήσητης Swap void swap( int *n1, int *n2) { int temp; temp = *n1; *n1 = *n2; *n2 = temp; } int x = 10, y=20; swap(&x,&y);

  13. 1 /* Fig. 7.15: fig07_15.c 2 This program puts values into an array, sorts the values into sort gets passed the address of array elements (pointers). The name of an array is a pointer. 3 ascending order, and prints the resulting array. */ 4 #include <stdio.h> 5 #define SIZE 10 6 void bubbleSort( int *, constint ); 7 8 int main() 9 { 10 11 int a[ SIZE ] = { 2, 6, 4, 8, 10, 12, 89, 68, 45, 37 }; 12 int i; 13 14 printf( "Data items in original order\n" ); 15 16 for ( i = 0; i < SIZE; i++ ) 17 printf( "%4d", a[ i ] ); 18 19 bubbleSort( a, SIZE ); /* sort the array */ 20 printf( "\nData items in ascending order\n" ); 21 22 for ( i = 0; i < SIZE; i++ ) 23 printf( "%4d", a[ i ] ); 24 25 printf( "\n" ); 26 27 return 0; 28 } 29 30 void bubbleSort( int *array, constint size ) 31 { 32 void swap( int *, int * );

  14. 33 int pass, j; 34 for ( pass = 0; pass < size - 1; pass++ ) 35 36 for ( j = 0; j < size - 1; j++ ) 37 38 if ( array[ j ] > array[ j + 1 ] ) 39 swap( &array[ j ], &array[ j + 1 ] ); 40 } 41 42 void swap( int *element1Ptr, int *element2Ptr ) 43 { 44 int hold = *element1Ptr; 45 *element1Ptr = *element2Ptr; 46 *element2Ptr = hold; 47 } Data items in original order 2 6 4 8 10 12 89 68 45 37 Data items in ascending order 2 4 6 8 10 12 37 45

  15. Αριθμητική Δεικτών • Στους δείκτες μπορούμε να πραγματοποιούμε αριθμητικές πράξεις: • ++ή–- • Ο δείκτης δείχνει στο επόμενο ή προηγούμενο αντικείμενο, δηλ. αν είναι δείκτης σε ακέραιο στον επόμενο ή προηγούμενο ακέραιο • Αντίστοιχα για +ή+= , -ή-= • Οι δείκτες μπορούν να αφαιρούνται ο ένας από τον άλλο • Όλες αυτές οι πράξεις δεν έχουν νόημα εκτός αν πραγματοποιούνται πάνω δείκτη που δείχνει στα στοιχεία ενός πίνακα

  16. Αριθμητική Δεικτών location 3000 3004 3008 3012 3016 pointer variable vPtr v[0] v[1] v[2] v[4] v[3] • Πίνακας 5 στοιχείων τύπουintσε μηχανή με 4 byte ints • int v[5]; • int *vPtr; • vPtr= &v[0] ή vPtr= v; //δείχνει στο πρώτο στοιχείοv[0] //vPtrστο3000 • vPtr += 2; //vPtrστο3008 • vPtrδείχνει στοv[ 2 ] (αύξηση κατά 2), μια καιη μηχανή έχει 4 byte ints, οπότε δείχνει3008

  17. Σχέση μεταξύ πινάκων και δεικτών • Στοιχείοb[ 3 ] • Μπορούμε να το πάρουμε σαν*( bPtr + 3 ) • ΣανbPtr[ 3 ] • bPtr[ 3 ]==b[ 3 ] • Με αριθμητική δεικτών: *( b + 3 )

  18. Διεύθυνση Πινάκων και Δείκτες int *a; a δείκτης σε ακέραιο Δεσμεύεται μνήμη μόνο για την αποθήκευση του a (της διεύθυνσης) Μπορούμε να αλλάξουμε την διεύθυνση που δείχνει ο a int b[10]; b σταθερή διεύθυνση (δείκτης) σε ακέραιο Δεσμεύεται μνήμη για 10 ακέραιους Δεν μπορούμε να αλλάξουμε την διεύθυνση στην οποία αντιστοιχεί ο b

  19. Πίνακες και Δείκτες • Μπορούμε να περνάμε δείκτες σε συναρτήσεις που περιμένουν πίνακεςκαι αντίστροφα. Σε κάθε περίπτωση περνάμε απλώς μία διεύθυνση μνήμης. Το παρακάτω γίνεται δεκτό: void f(int a[]); void q(int *b); int main() { int *c; int d[10]; f(c); q(d) }

  20. Παράδειγμα - ptrarithm2.c (Ι) int n[10] = {2, 3, 4, 5, 6, 7, 8, 9, 10, 1}, *p, c; // First way to traverse the array printf("First way to index the array.\n"); for (c=0; c < 10; c++) printf("n[%d] = %d\n", c, n[c]); printf("\nSecond way to index the array, through pointer arithmetic.\n"); for (c=0; c < 10; c++) printf("n[%d] = %d\n", c, *(n+c));

  21. Παράδειγμα - ptrarithm2.c (ΙΙ) printf("\nThird way to index the array, through pointers arithmetic.\n"); // We set the pointer to the beginning of the array p = n; for (c=0; c < 10; c++) printf("n[%d] = %d\n", c, *(p+c)); printf("\nFourth way to index the array, through pointers arithmetic.\n"); p = n; for (c=0 ; c < 10; c++) printf("n[%d] = %d, %d\n", c, *(p++), p);

More Related