This shows you the differences between two versions of the page.
|
java:generics [2017/02/07 14:37] gthanos [Interfaces ως παραμετρικοί τύποι δεδομένων] |
java:generics [2020/03/10 09:44] |
||
|---|---|---|---|
| Line 1: | Line 1: | ||
| - | ====== Παραμετρικοί τύποι δεδομένων (Generics) ====== | ||
| - | |||
| - | Ένας //generic// (γενικός) τύπος δεδομένων είναι ένας τύπος δεδομένων ο οποίος μπορεί να λαμβάνει ως παραμέτρους άλλους τύπους δεδομένων. Γενικότερα όταν θέλουμε να χρησιμοποιήσουμε κλάσεις που μπορούν να αποθηκεύσουν αντικείμενα οποιασδήποτε κλάσης (π.χ. για αν δημιουργήσουμε μία διασυνδεδεμένη λίστα) τότε θα πρέπει τις κλάσεις αυτές να τις κάνουμε τόσο γενικές ώστε να μπορούν να λαμβάνουν ως όρισμα οποιονδήποτε τύπο δεδομένων. Η υιοθέτηση γενικών κλάσεων έχει το πλεονέκτημα ότι μπορεί να αποθηκεύσει οποιαδήποτε κλάση, όμως απαιτεί αρκετά type-casts και ενέχει κινδύνους ως προς την ορθή διαχείριση των δεδομένων από τους προγραμματιστές. | ||
| - | |||
| - | Δείτε το παρακάτω παράδειγμα της κλάσης Box, η οποία θέλουμε να μπορεί να αποθηκεύσει οποιονδήποτε τύπο δεδομένων. | ||
| - | |||
| - | <code java Box.java> | ||
| - | public class Box { | ||
| - | private Object object; | ||
| - | |||
| - | public void set(Object object) { this.object = object; } | ||
| - | public Object get() { return object; } | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | Στο παραπάνω παράδειγμα επειδή όλες οι κλάσεις στη Java κληρονομούν την κλάση Object, μπορείτε να περάσετε ως όρισμα στις μεθόδους της συγκεκριμένης κλάσης οποιοδήποτε τύπο δεδομένων. Δείτε το παρακάτω παράδειγμα κώδικα που χρησιμοποιεί την κλάση Box. | ||
| - | |||
| - | <code java BoxUsage.java> | ||
| - | public class BoxUsage { | ||
| - | | ||
| - | Box b = new Box(); | ||
| - | | ||
| - | | ||
| - | | ||
| - | } | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | Ο παραπάνω κώδικας αφού μεταγλωττιστεί και επιχειρήσουμε να το τρέξουμε παράγει το παρακάτω exception καθώς επιχειρούμε να αναθέσουμε μία μεταβλητή τύπου String ένα αντικείμενο τύπου Integer. | ||
| - | < | ||
| - | Exception in thread " | ||
| - | at BoxUsage.main(BoxUsage.java: | ||
| - | </ | ||
| - | |||
| - | Παρατηρήστε επίσης, | ||
| - | |||
| - | <code java Box.java> | ||
| - | /** | ||
| - | * Generic version of the Box class. | ||
| - | * @param <T> the type of the value being boxed | ||
| - | */ | ||
| - | public class Box< | ||
| - | // T stands for " | ||
| - | private T t; | ||
| - | |||
| - | public void set(T t) { this.t = t; } | ||
| - | public T get() { return t; } | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | Η παραπάνω δήλωση της κλάσης //Box// σημαίνει ότι κατά τον ορισμό αντικειμένων της κλάσης αυτά θα πρέπει να προσδιορίζεται ανάμεσα στους χαρακτήρες '<', | ||
| - | <code java> | ||
| - | Box< | ||
| - | Box< | ||
| - | Box< | ||
| - | </ | ||
| - | Ο τύπος δεδομένων που χρησιμοποιούμε ανάμεσα στους χαρακτήρες '<', | ||
| - | |||
| - | ===== Ονοματολογία παραμέτρων ===== | ||
| - | |||
| - | * E - Element (χρησιμοποείται στο Java Collections Framework) | ||
| - | * K - Key | ||
| - | * N - Number | ||
| - | * T - Type | ||
| - | * V - Value | ||
| - | * S,U,V etc. - 2nd, 3rd, 4th types | ||
| - | |||
| - | ===== Δημιουργία και χρήση αντικειμένων ενός παραμετρικού τύπου δεδομένων ===== | ||
| - | |||
| - | <code java> | ||
| - | Box< | ||
| - | </ | ||
| - | |||
| - | Εναλλακτικά μπορείτε να γράψετε | ||
| - | <code java> | ||
| - | Box< | ||
| - | </ | ||
| - | |||
| - | Εδώ ο compiler αντιλαμβάνεται ότι δημιουργείται μία μεταβλητή τύπου '' | ||
| - | |||
| - | **Προσοχή: | ||
| - | <code java> | ||
| - | Box< | ||
| - | </ | ||
| - | |||
| - | Στο τελευταίο ο compiler θα εκδώσει το παρακάτω warning. | ||
| - | < | ||
| - | Note: BoxUsage.java uses unchecked or unsafe operations. | ||
| - | Note: Recompile with -Xlint: | ||
| - | </ | ||
| - | |||
| - | ===== Interfaces ως παραμετρικοί τύποι δεδομένων ===== | ||
| - | |||
| - | Εκτός από κλάσεις μπορούμε να ορίσουμε και διεπαφές ως παραμετρικούς τύπους. Το interface Stack που ορίζεται παρακάτω είναι ένα παράδειγμα παραμετρικού interface το οποίο προδιαγράφει τη λειτουργία της στοίβας. | ||
| - | |||
| - | <code java Stack.java> | ||
| - | interface Stack <T> { | ||
| - | public int size(); | ||
| - | public boolean isEmpty(); | ||
| - | public void push(T obj); | ||
| - | public T pop(); | ||
| - | public T top(); | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | Το interface δηλώνει ότι η στοίβα λαμβάνει ως όρισμα ένα τύπο δεδομένων που αντιστοιχεί στα αντικείμενα που αποθηκεύονται σε αυτή. | ||
| - | |||
| - | Άλλο παράδειγμα παραμετρικού interface είναι το [[https:// | ||
| - | |||
| - | <code java> | ||
| - | interface Comparable< | ||
| - | int compareTo(T o) | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | |||
| - | ===== Παραμετρικοί τύποι δεδομένων με πολλές παραμέτρους ===== | ||
| - | |||
| - | Δείτε το παρακάτω παράδειγμα, | ||
| - | |||
| - | <code java Pair.java> | ||
| - | public interface Pair<K, V> { | ||
| - | public K getKey(); | ||
| - | public V getValue(); | ||
| - | public void setKey(K key); | ||
| - | public void setValue(V value); | ||
| - | } | ||
| - | </ | ||
| - | <code java OrderedPair.java> | ||
| - | public class OrderedPair< | ||
| - | |||
| - | private K key; | ||
| - | private V value; | ||
| - | |||
| - | public OrderedPair(K key, V value) { | ||
| - | this.key = key; | ||
| - | this.value = value; | ||
| - | } | ||
| - | |||
| - | public void setKey(K key) { this.key = key; } | ||
| - | public void setValue(V value) { this.value = value; } | ||
| - | public K getKey() | ||
| - | public V getValue() { return value; } | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | Με βάση τον παραπάνω κώδικα μπορείτε να δημιουργήσετε αντικείμενα επιμέρους τύπων ως εξής: | ||
| - | <code java OrderedPairUsage.java> | ||
| - | public class OrderedPairUsage { | ||
| - | public static void main(String args[]) { | ||
| - | Pair< | ||
| - | Pair< | ||
| - | OrderedPair< | ||
| - | // the following is not allowed | ||
| - | Pair< | ||
| - | } | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | ===== Απλοί παραμετρικοί τύποι δεδομένων (Raw Generic Types) ===== | ||
| - | |||
| - | Αντικείμενα της κλάσης Box ορίζονται συνήθως σε συνδυασμό με ένα reference τύπο δεδομένων, | ||
| - | <code java> | ||
| - | Box b = new Box() | ||
| - | </ | ||
| - | |||
| - | Σε αυτή την περίπτωση καταλήγουμε να έχουμε έναν μη-παραμετρικό τύπο δεδομένων. Ο compiler δεν είναι σε θέση να γνωρίζει τον τύπο δεδομένων που θα τοποθετήσετε στο συγκεκριμένο αντικείμενο. Ουσιαστικά η παραπάνω δήλωση ισοδυναμεί με μία δήλωση της μορφής | ||
| - | <code java> | ||
| - | Box< | ||
| - | </ | ||
| - | |||
| - | Η προσπάθεια μεταγλώττισης του παρακάτω κώδικα καταδεικνύει το παραπάνω | ||
| - | <code java BoxUsage.java> | ||
| - | public class BoxUsage { | ||
| - | | ||
| - | Box b = new Box(5); | ||
| - | | ||
| - | | ||
| - | } | ||
| - | } | ||
| - | </ | ||
| - | |||
| - | Αν αλλάξετε την γραμμή '' | ||
| - | |||
| - | Η μη χρήση των παραμετρικών τύπων δεδομένων δυσκολεύει τον compiler να αποφασίσει για την ασφαλή ή μη ασφαλή χρήση των δεδομένων που θα αποθηκευτούν στην κλάση. Δείτε τα παρακάτω παραδείγματα κώδικα | ||
| - | |||
| - | <code java> | ||
| - | Box< | ||
| - | Box rawBox = stringBox; | ||
| - | </ | ||
| - | |||
| - | <code java> | ||
| - | Box rawBox = new Box(); | ||
| - | Box< | ||
| - | </ | ||
| - | |||
| - | <code java> | ||
| - | Box< | ||
| - | Box rawBox = stringBox; | ||
| - | rawBox.set(8); | ||
| - | </ | ||
| - | |||
| - | |||
| - | ==== Unchecked Error Messages ==== | ||
| - | |||
| - | Όπως είπαμε προηγούμενα η μίξη generics με raw types μπορεί να δημιουργήσει warnings κατά την μεταγλώττιση της μορφής | ||
| - | < | ||
| - | Note: Example.java uses unchecked or unsafe operations. | ||
| - | Note: Recompile with -Xlint: | ||
| - | </ | ||
| - | όπως παρακάτω | ||
| - | <code java WarningDemo.java> | ||
| - | public class WarningDemo { | ||
| - | public static void main(String[] args){ | ||
| - | Box< | ||
| - | bi = createBox(); | ||
| - | } | ||
| - | |||
| - | static Box createBox(){ | ||
| - | return new Box(); | ||
| - | } | ||
| - | } | ||
| - | </ | ||
| - | Μεταγλωττίζοντας με το flag '' | ||
| - | < | ||
| - | WarningDemo.java: | ||
| - | found : Box | ||
| - | required: Box< | ||
| - | bi = createBox(); | ||
| - | ^ | ||
| - | 1 warning | ||
| - | </ | ||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||
| - | |||