inetus3r Αύγουστος 9, 2012 #21 Αύγουστος 9, 2012 Αν μου επιτρέπεις την παρατήρηση, πάλι γίνεσαι απόλυτος:δεν υπάρχει κανένα (μα κανένα) πλεονέκτημαΤα στοιχεία μιας γλώσσας προγραμματισμού (πχ variables) υπάρχουν για να εξυπηρετήσουν ανάγκες.Όπως αναφέρεις κι εσύ, η κακή χρήση τους από τον ανθρώπινο παράγοντα είναι αυτή που δημιουργεί τα bugs, και όχι τα ίδια τα στοιχεία.Το παραπάνω (δηλ. η κακή χρήση) ισχύει βέβαια και για άλλα επιτεύγματα του ανθρώπινου πολιτισμού.
dop Αύγουστος 9, 2012 #22 Αύγουστος 9, 2012 δεν υπάρχει κανένα (μα κανένα) πλεονέκτημα για global/static μεταβλητές στους σύγχρονους compilers όταν υπάρχουν εναλλακτικέςΌταν κάνεις παράθεση κάτι, πρέπει να την κάνεις χωρίς να αλλοιώνεις το νόημα. Ναι, υπάρχουν περιπτώσεις που αυτού του είδους οι μεταβλητές είναι χρήσιμες και ο μοναδικός τρόπος να υλοποιήσεις κάτι. Όχι, η περίπτωση του OP δεν είναι μία από αυτές. Το ίδιο ισχύει για όλα τα χαρακτηριστικά μιας γλώσσας.
inetus3r Αύγουστος 9, 2012 #23 Αύγουστος 9, 2012 Ακόμα κι αν φάνηκε έτσι, δεν είχα πρόθεση να αλλοιώσω το νόημα της φράσης σου.
Jaco Αύγουστος 9, 2012 #24 Αύγουστος 9, 2012 Όλες αυτές οι δυνατότητες δεν υπάρχουν τυχαία στους compilers, το ότι δεν μπορούν οι περισσότεροι να γράψουν κώδικα, δεν σημαίνει ότι είναι κακές πρακτικές... Οι static μεταβλητές είναι εξαιρετικά χρήσιμες και οι global το ίδιο και ειδικά με αυτές μπορείς να γράψεις optimized κώδικα από άποψη ταχύτητας...Ακόμα και η goto που λανθασμένα λένε ότι δεν πρέπει να χρησιμοποιείται, λύνει έξυπνα πολλά προβλήματα και μπορεί να κάνει πολύ πιο optimized κώδικα απ' ότι να γράφεις ένα σωρό κώδικα για να αποφύγεις την χρήση της...Οι κακές πρακτικές προγραμματισμού ευθύνονται όχι, τα εργαλεία της γλώσσας... δεν υπάρχουν λάθος εργαλεία, μόνο λάθος χρήση τους...
dop Αύγουστος 9, 2012 #25 Αύγουστος 9, 2012 Δύο απλά παραδείγματα:/* test_g.c */#include <stdio.h>int i = 0;void foo(void) { ++i; }int main(void){ foo(); printf("%d\n", i); return 0;}και/* test_ng.c */#include <stdio.h>void foo(int* j) { ++(*j); }int main(void){ int i = 0; foo(&i); printf("%d\n", i); return 0;}Βήμα δεύτερο:$ gcc --versiongcc (GCC) 4.7.1 20120614 (XXXX)Copyright (C) 2012 Free Software Foundation, Inc.This is free software; see the source for copying conditions. There is NOwarranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.$ gcc -O3 -Wall -S test_g.c -o test_g.s ; gcc -O3 -Wall -S test_ng.c -o test_ng.s ; diff test_g.s test_ng.s1c1< .file "test_g.c"---> .file "test_ng.c"8c8< addl $1, i(%rip)---> addl $1, (%rdi)23c23< movl i(%rip), %esi---> movl $1, %esi26,27d25< addl $1, %esi< movl %esi, i(%rip)35,41d32< .globl i< .bss< .align 16< .type i, @object< .size i, 4< i:< .zero 4Με εξαιρετικά απλό παράδειγμα, ήδη έχεις μια addl και μία movl παραπάνω. Γιατί οι global και static είναι στο data segment του προγραμματός σου.Και αν δεν αρκεί αυτό σαν παράδειγμα, σε μεγαλύτερους κώδικες, οι global / static μεταβλητές θα σου κλείσουν τις περισσότερες φορές το vectorization, inlining, constant propagation, loop fusion/coallescing κλπ (εκτός και αν είναι read-only). Και αυτό μπορεί να σημαίνει μέχρι και 3x πιο αργό εκτελέσιμο αν τύχει οι χαμένες βελτιστοιποιήσεις να είναι στο κρίσιμο μονοπάτι του κώδικα.Για να βάλω και το OpenMP / threads στο παιχνίδι, η static / global θέλει ένα critical section γύρω της, ώστε να αποφευχθούν data races. Αυτό σημαίνει ότι μόλις δημιουργήθηκε ένα thread contention point - πάει το scalability.Τέλος, τις περισσότερες φορές οι global / static θα προκαλέσουν και cache misses. Αν ακόμα επιμένετε ότι είναι καλή ιδέα να χρησιμοποιείς global / static, μου τελείωσαν τα επιχειρήματα.ΥΓ όπως είπα παραπάνω, κάποια προβλήματα λύνονται μόνον με global μεταβλητές. Αλλά τότε τις κάνεις και static ώστε να χρησιμοποιούνται μόνον από το συγκεκριμένο compilation unit, ώστε να έχεις τα optimizations σου και να ελαχιστοποιείς τα cache misses.
inetus3r Αύγουστος 10, 2012 #26 Αύγουστος 10, 2012 Άρα, σύμφωνα με το ΥΓ σου, τελικά οι global μεταβλητές δεν είναι κακές (αφού λύνουν προβλήματα), εφόσον ξέρεις πότε να τις χρησιμοποείς και πως να τις αξιοποιείς.
dop Αύγουστος 10, 2012 #27 Αύγουστος 10, 2012 Το είπα παραπάνω:Ναι, υπάρχουν περιπτώσεις που αυτού του είδους οι μεταβλητές είναι χρήσιμες και ο μοναδικός τρόπος να υλοποιήσεις κάτι. Όχι, η περίπτωση του OP δεν είναι μία από αυτές. Το ίδιο ισχύει για όλα τα χαρακτηριστικά μιας γλώσσας.Τίποτα δεν είναι κακό. Αλλά τα πάντα δεν είναι καρφί επειδή ξέρεις να χρησιμοποιείς μόνον το σφυρί.
inetus3r Αύγουστος 11, 2012 #28 Αύγουστος 11, 2012 Συμφωνώ απόλυτα με το παραπάνω (και όχι με το 1 του post #11).Μια και αρχίσαμε αυτή τη κουβέντα, κι επειδή βρίσκομαι μακρυά από το χώρο μου, μπορείς να δεις το assembly output που θα προκύψει, αν στο post #25 αλλάξεις την θέση του ++ operator στις συναρτήσεις foo ? Έχεις την δυνατότητα να δοκιμάσεις κι άλλον compiler εκτός του gcc ?
dop Αύγουστος 11, 2012 #29 Αύγουστος 11, 2012 Ακριβώς το ίδιο αποτέλεσμα αν χρησιμοποιείς post-increment - ούτως ή άλλως δε υπάρχει ανάθεση, οπότε το αποτέλεσμα είναι το ίδιο.Ο clang 3.0 παράγει παραπλήσιο κώδικα με τον gcc. Μόνον ο intel κατάφερε να παράξει το ίδιο εκτελέσιμο και στις δύο περιπτώσεις. Άμα το θυμηθώ, Δευτέρα μπορώ να χρησιμοποίσω τον Cray compiler και PGI.
inetus3r Αύγουστος 11, 2012 #30 Αύγουστος 11, 2012 Η δοκιμή που έκανες με κάλυψε.Ευχαριστώ για την συνομιλία.
Recommended Posts
Archived
This topic is now archived and is closed to further replies.