Cours d'initiation à Python
Modularisation
G. Chagnon
Ce chapitre présente les bases de la programmation orientée objet en Python
python, initiation introduction, module, import, classe, objet
Il est tout à fait possible de se créer des bibliothèques de fonctions pour éviter de dupliquer un travail déjà fait dans le cadre d'un projet, ou bien pour pouvoir réutiliser des fonctions ou des définitions de classes (voir ci-après) dans des projets différents.
Pour cela, il faut simplement écrire la liste des fonctions dans un fichier .py
(par exemple mes_fonctions.py) puis importer ce fichier avant toute autre opération dans le code, avec les mots-clefs from... import
(dans ce cas, from mes_fonctions import *
). Cette syntaxe comporte plusieurs variantes :
from mes_fonctions import *
importe telles quelles toutes les fonctions définies dans le fichier
from mes_fonctions import fonction1, fonction2
importe uniquement les fonctions fonction1
et fonction2
Afin d'éviter d'éventuels conflits de noms (par exemple quand deux fonctions portent le même nom dans deux fichiers différents), d'une manière similaire aux espaces de noms XML
, on peut expliciter un préfixe pour le module importé :
import mes_fonctions
. Dans ce cas, les noms des fonctions importées doivent être préfixés par mes_fonctions
: mes_fonctions.fonction1
, mes_fonctions.fonction2
...
import mes_fonctions as mesFonc
. Dans ce cas, les noms des fonctions importées doivent être préfixés par l'alias mesFonc
, qui peut être défini indépendamment du nom du fichier : mesFonc.fonction1
, mesFonc.fonction2
...
Il n'est pas dans le propos de ce cours de faire une initiation complète à la Programmation Orientée Objet, mais d'en présenter les quelques points qui sont utiles au jour le jour en Python
.
Un objet est une entité de programmation qui possèdent un certain nombre de propriétés, et sur laquelle il est possible d'agir ou qui est susceptible d'agir par l'intermédiaire de méthodes. Une méthode est une sorte de fonction s'appliquant à l'objet, et à la syntaxe un peu particulière.
Une propriété caractérise un objet. Par exemple, si l'on pense à un objet physique comme une boîte, ses propriétés pourraient être ses longueur, hauteur et profondeur, sa couleur, le fait qu'elle soit ouverte ou fermée, pleine ou vide, etc. Des méthodes permettant d'interagir avec cette boîte seraient par exemple l'action de l'ouvrir ou de la fermer (qui agirait sur la propriété « ouverte ou fermée »), celle d'y ajouter ou d'en extraire du contenu (qui agirait sur la propriété « pleine ou vide »), etc. En Python
comme dans les autres langages orientés objet, on représente les propriétés et les méthodes avec des points. Pour reprendre notre exemple de boîte, une implémentation Python
en serait :
contenu
Une étape supplémentaire de conceptualisation d'un objet passe par la définition d'une classe. Une classe est une représentation abstraite d'une entité ; en tant que telle, elle n'a pas d'existence mais sert à définir des objets, qui en sont des instances. Par exemple, dans le monde physique on peut définir une classe que l'on nommera Gant
. Cette classe, conçue pour pouvoir décrire n'importe quel type de livre, possède des méthodes et des propriétés, par exemple une méthode retourner()
, et des propriétés couleur
, matière
et sens
. Mais le concept de gant en lui-même n'a pas de couleur ni de matière ; seul a une couleur un gant particulier, qui sera fabriqué en laine par exemple. Dans ce cas, on appelle classe la définition de toutes les propriétés et méthodes qu'un gant doit avoir, et instanciation les propriétés et méthodes d'un gant en particulier.
La définition d'une classe est tout à fait simulaire à celle d'une fonction. Par convention, on a l'habitude de mettre une majuscule à la première lettre du nom d'une classe.
ExempleClasse:
"""Documentation de la classe"""
Définition de la classe
p = ExempleClasse()
Les """
après le nom de la classe servent à la documentation. p
est une instanciation de la classe. Attention à cette étape à la duplication d'objets :
p1=ExempleClasse()
p2=ExempleClasse()
p3=p1
p1
<__main__.ExempleClasse object at 0x7ff404091ba8>
p2
<__main__.ExempleClasse object at 0x7ff404091b70>
p3
<__main__.ExempleClasse object at 0x7ff404091ba8>
On remarque dans le code précédent que les variables p1
et p3
renvoient bien au même objet, à l'adresse mémoire 0x7ff404091ba8.
Pour le moment, une telle classe est vide, et les objets qui l'instancient aussi.
On peut ensuite ajouter des propriétés à l'objet ainsi créé. On définit ainsi une nouvelle propriété x
:
p1.x="valeur de x"
La définition de méthodes se fait en revanche lors de la définition de la classe, et de manière similaire à une définition de fonction, à un détail près :
ExempleClasse:
methode1 (self, a,b):
self.x=2*a
self.y=2*a*b
Le mot-clef self
est en effet obligatoire et doit apparaître comme premier argument.
On peut ensuite instancier la classe ExempleClasse
, puis utiliser la méthode methode1
pour créer les deux propriétés x
et y
de l'objet instancié :
test = ExempleClasse()
test.methode1(2,3)
test.x, test.y
L'exemple précédent affiche 4 et 12.
On peut d'ailleurs tout à fait définir des valeurs par défaut, de la même manière que pour les fonctions :
exempleClasse:
methode1(self, a=3,b=8):
self.x=2*a
self.y=2*a*b
En programmation orientée objet, le constructeur permet de « construire » un nouvel objet à partir d'une classe, en lui passant des paramètres. Jusqu'à présent en effet, quand nous écrivions nouvelObjet = ExempleClasse()
, le nouvel objet ne possédait pas de propriété prédéfinie. Un constructeur le permet.
En Python
, le constructeur est une méthode portant un nom prédéfini et réservé, __init__
(commençant et finissant par deux « underscores »). Comme toute méthode, le constructeur peut accepter des paramètres, avec ou sans valeur par défaut. Par exemple…
Test:
__init__ (self, t=4, z=5):
self.x="varx"
self.y="vary"
self.t=t
self.z=z
nouvelObjet0=Test()
nouvelObjet1=Test(z=3)
nouvelObjet0.x, nouvelObjet0.y, nouvelObjet0.t, nouvelObjet0.z
nouvelObjet1.x, nouvelObjet1.y, nouvelObjet1.t, nouvelObjet1.z
Ce code affiche varx vary 4 5
puis varx vary 4 3
.
L'encapsulation consiste en le masquage de certaines propriétés afin qu'elles ne soient plus directement accessibles depuis l'extérieur de l'objet. Pour définir une propriété privée, on préfixe son nom, encore une fois, par deux « underscores » dans le constructeur :
ClasseTest :
Constructeur
__init__ (self):
self.a=1
self.b=2
self.__c=3
self.__d=4
getters
get_a(self):
self.a
get_b(self):
self.b
get_c(self):
self.__c
get_d(self):
self.__d
setters
set_a(self, x):
self.a=x
set_b(self, x):
self.b=x
set_c(self, x):
self.__c=x
set_d(self, x):
self.__d=x
test=ClasseTest()
Dans l'exemple précédent, les propriétés test.a
et test.b
sont accessibles directement, mais print(test.c)
renvoie un message d'erreur, car la propriété est introuvable.
En programmation orientée objet, l'héritage est la possibilité pour une classe de dériver d'une classe déjà existante, en en récupérant propriétés et méthodes, tout en lui ajoutant d'autres. Pour cela, il suffit d'indiquer :
- au moment de la défnition du nom de la classe enfant, le nom de sa classe parente :
class NomDeLaClasseEnfant(NomDeLaClasseParente)
- dans le constructeur de la classe enfant la ligne
NomDeLaClasseParente.__init__(self)
Par exemple :
Parente :
"""Classe parente"""
__init__(self):
self.__prop1=1
self.__prop2=2
get_prop1(self):
self.__prop1
get_prop2(self):
self.__prop2
set_prop1(self, x):
self.__prop1=x
set_prop2(self, x):
self.__prop2=x
Enfant(Parente) :
"""Classe enfant"""
__init__(self):
Parente.__init__(self)
self.__prop3=3
get_prop3(self):
self.__prop3
set_prop3(self, x):
self.__prop3=x