I. Introduction▲
Introduits par la PEP 318, les décorateurs (decorator en anglais) ne sont ni plus ni moins que des fonctions permettant de modifier le comportement d'autres fonctions et/ou d'exécuter du code supplémentaire.
Ainsi, plutôt que de copier/coller bêtement du code afin de légèrement modifier le comportement d'une fonction, Python vous propose d'éviter la répétition de code, réduisant ainsi les problèmes de maintenance du code via la possibilité de créer des fonctions d'ordre supérieur.
Les décorateurs sont considérés, en Python, comme une notion assez évoluée. La possibilité de passer en paramètre, à une fonction, une autre fonction, ne semble en effet pas forcément automatique ou naturelle en programmation procédurale ou objet.
Cependant, c'est ce que les décorateurs vont nous permettre de faire : via une fonction, manipuler d'autres fonctions (aussi bien en entrée qu'en sortie) et éventuellement impacter leur comportement. Pour ce faire, il suffira de passer le nom de la fonction à manipuler (sans les « () » ) comme n'importe quel autre paramètre.
Côté application, cela peut être fort utile pour avoir du code intelligent sachant s'adapter à divers environnements, générer des logs de l'exécution du code…
L'ensemble des codes de cet article sont écrits en Python 3.x. Par conséquent leur fonctionnement en Python 2.X n'est pas garanti. Pour rappel, Python 3.X est déclaré apte pour la production depuis la version 3.4 et il est fortement recommandé de ne plus utiliser que Python 3.X, sauf en cas de contrainte forte (bibliothèques non portées, serveur impossible à migrer…)
II. Exemple simplifié de décorateur▲
2.
3.
4.
5.
6.
7.
8.
9.
def
mon_decorateur
(
fonction):
print
(
fonction.__name__
+
' appelée'
)
return
fonction
(
)
def
ma_fonction
(
):
print
(
"hello_world"
)
if
__name__
==
"__main__"
:
mon_decorateur
(
ma_fonction)
Voici un exemple simplifié de décorateur. Dans le code ci-dessus, nous appelons notre fonction de décoration « mon_decorateur() » en lui passant en argument une fonction : ma_fonction(). Par rapport à la fonction seule (NDLR : ma_fonction), nous exécutons du code supplémentaire et par conséquent, modifions le comportement de la fonction. Nous sommes donc bien en présence d'un décorateur.
En réalité, nous ne modifions nullement le comportement du code de la fonction, puisque nous n'y touchons pas. C'est au niveau de l'appel de la fonction que tout se joue et ce sera le résultat final qui sera impacté.
Cependant cela a l'inconvénient que vous deviez appeler le décorateur et non pas la fonction à chaque fois que vous avez besoin de cette fonctionnalité supplémentaire.
Python permet de simplifier cette écriture via une syntaxe dédiée aux décorateurs, que l'on déclare avec le « @ ».
2.
3.
4.
5.
6.
7.
8.
9.
10.
def
mon_decorateur
(
fonction):
print
(
fonction.__name__
+
' appelée'
)
return
fonction
@mon_decorateur
def
ma_fonction
(
):
print
(
"hello_world"
)
if
__name__
==
"__main__"
:
ma_fonction
(
)
Dans ce cas, nous voyons que nous appelons notre fonction comme si elle n'était pas décorée. Python se charge, de façon transparente pour nous, de lui substituer sa version décorée
Une question vous vient alors peut-être à l'esprit : et si on ne désire pas décorer notre fonction ? Eh bien il suffit de passer par une variable.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
a_decorer =
True
def
mon_decorateur
(
fonction):
if
a_decorer:
print
(
fonction.__name__
+
' appelée'
)
return
fonction
@mon_decorateur
def
ma_fonction
(
):
print
(
"hello_world"
)
if
__name__
==
"__main__"
:
ma_fonction
(
)
Et voilà. Si a_decorer vaut True, alors notre décorateur jouera son rôle. Mais si a_decorer vaut False, alors nous récupérons notre fonction standard. Pour passer d'une version de débogage qui requiert le log des appels de fonction à une version de distribution, il vous suffira de changer cette valeur.
C'est sur cela que repose le fonctionnement de base d'un décorateur.
Il est possible d'enchaîner les décorateurs. Ainsi faire
mon_decorateur01
(
mon_decorateur02
(
ma_fonction))
s'écrira
2.
3.
4.
@mon_decorateur01
@mon_decorateur02
ma_fonction
(
):
...
III. Les décorateurs en vrai▲
Ce que nous venons de voir précédemment est bien sympathique, néanmoins d'une utilité limitée. En effet, nous avons fait abstraction des potentiels arguments que nous voudrions passer à la fonction. Entrons dans le vif du sujet.
Dans sa version idiomatique, un décorateur n'exécutera jamais lui-même du code, même s'il le peut. En effet, il préférera créer une nouvelle fonction, décoration de la fonction passée en paramètre, et vous renverra la fonction décorée, laquelle sera alors interprétée par Python. C'est là le point important : un vrai décorateur renvoie toujours un « callable », c'est-à-dire une fonction.
Afin d'être plus clair, reprenons notre exemple précédent.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
def
mon_decorateur
(
fonction):
def
ma_fonction_decoree
(
):
print
(
fonction.__name__
+
' appelée'
)
fonction
(
)
return
ma_fonction_decoree
@mon_decorateur
def
ma_fonction
(
):
print
(
"hello_world"
)
if
__name__
==
"__main__"
:
ma_fonction
(
)
Basiquement, voici un véritable décorateur. « mon_decorateur » définit une fonction décorant la fonction passée en paramètre, puis renvoie cette fonction décorée.
Notez que le résultat est le même que précédemment. Maintenant continuons à monter en puissance, en décorant une fonction avec paramètres.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
def
mon_decorateur
(
fonction):
def
ma_fonction_decoree
(*
args, **
kwargs):
print
(
fonction.__name__
+
' appelée'
)
print
(
'arguments non nommes passes en parametres:
%s
'
%
(
args))
return
fonction
(*
args, **
kwargs)
return
ma_fonction_decoree
@mon_decorateur
def
ma_fonction
(
nom):
print
(
'hello_'
+
nom)
return
True
if
__name__
==
'__main__'
:
print
(
ma_fonction
(
'developpez.com'
))
Voici un exemple un peu plus proche de ce que vous serez amené à voir réellement. Les arguments, nommés ou non, sont attrapés par la fonction imbriquée dans notre décorateur, et transmis à la fonction appelée.
Dans cette fonction imbriquée, nous effectuons directement un « return » de la fonction appelée.
Notons également que dans cet exemple nous constatons que le décorateur a accès aux arguments passés en paramètres.
Voyons maintenant une petite particularité importante à connaître. Exécutez le code suivant.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
def
mon_decorateur
(
fonction):
print
(
"Instance de decorateur en memoire"
)
def
ma_fonction_decoree
(*
args, **
kwargs):
print
(
fonction.__name__
+
' appelée'
)
return
fonction
(*
args, **
kwargs)
return
ma_fonction_decoree
@mon_decorateur
def
ma_fonction
(
nom):
print
(
'hello_'
+
nom)
return
True
@mon_decorateur
def
ma_fonction2
(
website):
print
(
'Visitez-nous sur '
+
website)
if
__name__
==
'__main__'
:
print
(
ma_fonction
(
'developpez.com'
))
ma_fonction2
(
'http://www.developpez.com'
)
Que constatez-vous dans le terminal d'exécution ? Chaque « @mon_decorateur » débouche sur la mise en mémoire, en cache, d'une instance de notre décorateur. Cela est assimilable au concept de programmation fonctionnelle appelée mémoïsation.
Cela est fort utile, car vous pourrez alors utiliser cela à votre avantage pour compter le nombre d'appels des fonctions décorées par exemple.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
def
mon_decorateur
(
fonction):
counter =
0
print
(
"Instance de decorateur en memoire"
)
def
ma_fonction_decoree
(*
args, **
kwargs):
nonlocal counter
counter +=
1
print
(
"Appel N°:
%s
"
%
(
counter))
print
(
fonction.__name__
+
' appelée'
)
return
fonction
(*
args, **
kwargs)
return
ma_fonction_decoree
@mon_decorateur
def
ma_fonction
(
nom):
print
(
'hello_'
+
nom)
return
True
if
__name__
==
'__main__'
:
print
(
ma_fonction
(
'developpez.com'
))
print
(
ma_fonction
(
'developpez.com'
))
print
(
ma_fonction
(
'developpez.com'
))
« nonlocal » est une nouveauté de la branche 3. Ce nouveau mot clé ne fonctionne que dans les cas de plusieurs fonctions imbriquées. Il permet de stipuler que la fonction imbriquée a le droit, dans son espace de noms propre, de modifier les variables ainsi déclarées dans l'espace de noms de sa fonction mère. Voici un code permettant de mettre en évidence l'usage de ce mot clé.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
def
fonction
(
):
var1 =
0
var2 =
0
print
(
"Valeurs initiales"
)
print
(
"var1=
%d
"
%
(
var1))
print
(
"var2=
%d
"
%
(
var2))
def
demo
(
):
nonlocal var1
var1 +=
2
var2 =
var1 +
3
print
(
"*****************"
)
print
(
"*****************"
)
print
(
"var1=
%d
"
%
(
var1))
print
(
"var2=
%d
"
%
(
var2))
demo
(
)
print
(
"*****************"
)
print
(
"var1=
%d
"
%
(
var1))
print
(
"var2=
%d
"
%
(
var2))
if
__name__
==
'__main__'
:
fonction
(
)
On voit bien ici que la variable déclarée en « nonlocal » a bien été modifiée, mais pas la seconde variable.
IV. Exemple d'application concrète▲
L'exemple le plus basique que nous rencontrons le plus souvent est la mesure du temps d'exécution. Même s'il existe des bibliothèques dédiées, nous pouvons le faire facilement via un décorateur.
Reprenons notre exemple précédent.
2.
3.
4.
5.
6.
7.
8.
9.
10.
11.
12.
13.
14.
15.
16.
17.
18.
19.
20.
21.
22.
23.
24.
25.
26.
import
time
a_mesurer =
True
def
mon_decorateur
(
fonction):
if
a_mesurer:
def
mesure_execution
(
):
start_time =
time.time
(
)
print
(
'start_time:
%.3f
'
%
(
start_time))
result =
fonction
(
)
stop_time =
time.time
(
)
print
(
'stop_time:
%.3f
'
%
(
stop_time))
total =
stop_time -
start_time
print
(
"temps d'execution:
%.3f
"
%
(
total))
return
result
return
mesure_execution
else
:
return
fonction
@mon_decorateur
def
ma_fonction
(
):
print
(
'hello_world'
)
time.sleep
(
0.25
)
if
__name__
==
'__main__'
:
ma_fonction
(
)
Le plus grand piège ici est au niveau du décorateur. En effet, mal déclaré vous pourrez voir apparaître un certain nombre d'erreurs potentielles. Il ne faut pas oublier en effet qu'un décorateur doit toujours renvoyer une fonction, autrement dit un « callable ». Sans cela, une erreur sera signalée.
V. Conclusion▲
Comme nous venons de le voir, les décorateurs font partie des concepts Python qui peuvent rapidement se révéler fort utiles.
Très utilisés par les développeurs Python aguerris, les décorateurs permettent d'obtenir du code concis, limitant les répétitions au strict minimum, sans pour autant renoncer à une lisibilité importante.
J'espère que cette introduction aura su vous convaincre, et vous permettra à l'avenir d'être encore plus performant avec Python.