- Encapsulation et Extension des Types
- Héritage
- Polymorphisme
- Surcharge
- Fonctions virtuelles
- Membres Statiques de Classe
- Templates de fonctions
- Templates de classes
- Abstract Classes
Fonctions virtuelles
Le mot-clé virtual sert de spécificateur pour la fonction, et assure le mécanisme de choix dynamique lors de l'exécution de la fonction-membre qui convient (parmi les fonctions des classes de base et dérivée), les structures ne peuvent pas avoir de fonctions virtuelles. Il peut être utilisé pour changer les déclarations des membres de la fonction seulement.
La fonction virtuelle, comme une fonction ordinaire, doit avoir un corps exécutable. A l'appel, sa sémantique est exactement la même, comme dans les autres fonctions.
La fonction virtuelle peut être remplacée dans la classe dérivée. Le choix de quelle définition de fonction appeler pour une fonction virtuelle, est fait dynamiquement (à l'étape de l'exécution). Un cas typique est quand une classe de base contient une fonction virtuelle et que les classes dérivées ont leurs propres versions de cette fonction.
Le pointeur sur la classe de base peut indiquer l'objet de classe, ou l'objet d'une classe dérivée. Le choix de la fonction-membre appelée sera effectué à l'étape de l'exécution et dépendra du type de l'objet, et non du type du pointeur. S'il n'y a aucun membre d'un type dérivé, une fonction virtuelle de la classe de base est utilisée par défaut.
Les destructeurs sont toujours virtuels, indépendamment du fait qu'ils soient déclarés avec le mot-clé virtual ou non.
Considérons l'utilisation des fonctions virtuelles avec l'exemple du programme MT5_Tetris.mq5. La classe de base CTetrisShape avec une fonction virtuelle Draw (dessiner) est définie dans le fichier inclus MT5_TetrisShape.mqh.
//+------------------------------------------------------------------+
|
Ensuite, pour chaque classe dérivée, cette fonction est implémentée conformément aux particularités de la classe-descendante. Par exemple, la première figure CTetrisShape1 a son implémentation de fonction Draw() :
class CTetrisShape1 : public CTetrisShape
|
La figure "carré" est décrite par la classe CTetrisShape6 et elle a sa propre implémentation de méthode Draw() :
class CTetrisShape6 : public CTetrisShape
|
En fonction de l'objet, une fonction virtuelle de n'importe quelle classe dérivée est appelée.
void CTetrisField::NewShape()
|
Modifier 'override' #
Le modificateur 'override' signifie que la fonction déclarée doit dériver la méthode de sa classe parente. L'utilisation de ce mot-clé vous permet d'éviter des erreurs de dérivation, par exemple il vous permet d'éviter la modification accidentelle de la signature de la méthode. Supposons que la méthode 'func' est définie dans la classe de base. La méthode accepte une variable de type int comme argument :
class CFoo |
Ensuite, la méthode est dérivée dans la classe fille :
class CBar : public CFoo |
Cependant, le type de l'argument a été changé par erreur de int en short. De fait, ce n'est pas une dérivation de méthode mais une surcharge de méthode. Conformément à l'algorithme de définition des fonctions surchargées, le compilateur peut dans certains cas choisir une méthode définie dans la classe de base au lieu de la méthode dérivée.
Pour éviter ce genre d'erreurs, vous devriez ajouter explicitement le modificateur 'override' à la méthode que vous souhaitez dériver.
class CBar : public CFoo |
Si la signature de la métode est changée pendant la dérivation, le compilateur ne pourra pas trouver une méthode avec la même signature dans la classe parente, et il retournera une erreur de compilation :
La méthode 'CBar::func' est déclarée avec le spécificateur 'override' mais ne dérive d'aucune méthode de la classe de base |
Modificateur final #
Le modificateur 'final' effectue l'inverse il interdit tout dérivation de la méthode dans les classes filles. Si l'implémentation d'une méthode est suffisante et complète, déclarez cette méthode avec le modificateur 'final' pour être sûr qu'elle ne sera pas modifiée ultérieurement.
class CFoo |
Si vous essayez de dériver une méthode ayant le modificateur 'final' comme montré dans l'exemple ci-dessus, le compilateur retournera une erreur :
La méthode 'CFoo::func' déclarée comme 'final' ne peut pas être dérivée par 'CBar::func' |
Voir aussi