Python: Fonctions: Difference between revisions

From Wiki Cours
Jump to navigation Jump to search
Line 57: Line 57:


Dans ce qui suit, arguments et paramètres seront considérés comme synonymes.
Dans ce qui suit, arguments et paramètres seront considérés comme synonymes.
f.__module__ : affiche le module de rattachement d'une fonction
* f.__module__ : affiche le module de rattachement d'une fonction
f.__name__ : affiche le nom de la fonction, par exemple si g = f, g.__name__ donne 'f'
* f.__name__ : affiche le nom de la fonction, par exemple si g = f, g.__name__ donne 'f'


Signature d'une fonction: c'est la combinaison du nom et de la liste d'arguments
'''Signature d'une fonction''': c'est la combinaison du nom et de la liste d'arguments
<source lang="py">
<source lang="py">
def f(a,b,c):  
def f(a,b,c):  
     return a+b**2+c**3
     return a+b**2+c**3
f(1,2,3)
print f.__module__
g = f
print g.__name__
</source>
</source>
définit localement les variables a,b,c qui prendront comme valeurs celles transmises lors de l'appel de la fonction, par exemple f(1,2,3). Il est obligatoire de donner la liste des arguments et dans le bon ordre.
définit localement les variables a,b,c qui prendront comme valeurs celles transmises lors de l'appel de la fonction, par exemple f(1,2,3). Il est obligatoire de donner la liste des arguments et dans le bon ordre.


arguments optionnels: il est possible de mettre des arguments optionnels à condition de leur donner une valeur par défaut et ils doivent nécessairement être placés après les arguments obligatoires
'''Arguments optionnels''': il est possible de mettre des arguments optionnels à condition de leur donner une valeur par défaut et ils doivent nécessairement être placés après les arguments obligatoires:
def f(a,b,c=0): c est optionnel donc f(1,2) fonctionne et c prendra la valeur 0
f.__defaults__ : affiche les valeurs par défaut des arguments optionnels
<source lang="py">
<source lang="py">
def f(a,b,c=0):  # c est optionnel donc f(1,2) fonctionne et c prendra la valeur 0
    return a+b**2+c**3
print f.__defaults__ # affiche les valeurs par défaut des arguments optionnels
def f(a=1,b=2,c=3):
def f(a=1,b=2,c=3):
     return a+b**2+c**3
     return a+b**2+c**3
Line 77: Line 82:
</source>
</source>


paramètres nommés: lorsqu'il y a des paramètres optionnels, on peut ne passer que l'argument dont on veut attribuer une valeur différente de la valeur par défaut, sans avoir à respecter l'ordre:
'''Arguments nommés''': lorsqu'il y a des paramètres optionnels, on peut ne passer que l'argument dont on veut attribuer une valeur différente de la valeur par défaut, sans avoir à respecter l'ordre:
<source lang="py">
<source lang="py">
f(b=3)
f(b=3)
f(c=2,a=2)
f(c=2,a=2)
</source>
</source>
On parle de paramètres nommés. Cela donne une lecture claire de l'appel à la fonction. Si l'on utilise des paramètres non nommés et des paramètres nommés, ces derniers sont toujours placés à la fin:
On parle de paramètres nommés. Cela donne une lecture claire de l'appel à la fonction. Si l'on utilise des arguments non nommés et des arguments nommés, ces derniers sont toujours placés à la fin:
<source lang="py">
<source lang="py">
f(1,c=3)
f(1,c=3)
</source>
</source>


paramètres extensibles non nommés: Python permet à une fonction d'accepter une liste extensible d'arguments, c'est-à-dire que le nombre d'arguments n'est pas défini à l'avance. Il faudra dans ce cas veiller à ce que la fonction gère correctement cette liste quelconque d'arguments. La syntaxe est la suivante, avec une '''*''' devant le nom de ce qui sera un tuple contenant la liste des paramètres (on aime bien utiliser 'args' mais on peut appeler ce tuple autrement):
'''Liste extensible d'arguments non nommés''': Python permet à une fonction d'accepter une liste extensible d'arguments, c'est-à-dire que le nombre d'arguments n'est pas défini à l'avance. Il faudra dans ce cas veiller à ce que la fonction gère correctement cette liste quelconque d'arguments. La syntaxe est la suivante, avec une '''*''' devant le nom de ce qui sera un '''tuple''' contenant la liste des paramètres (on aime bien utiliser 'args' mais on peut appeler ce tuple autrement):
<source lang="py">
<source lang="py">
def f(*args):
def f(*args):
Line 100: Line 105:
</source>
</source>


paramètres extensibles nommés: il est en fait plus judicieux d'utiliser une liste de paramètres nommés extensible. La syntaxe est la suivante, avec cette fois une double étoile '''**''' et kwargs n'est alors plus un tuple mais un dictionnaire associant nom et valeur des variables à transmettre:
'''Liste extensible d'arguments nommés''': il est en fait plus judicieux d'utiliser une liste de paramètres nommés extensible. La syntaxe est la suivante, avec cette fois une double étoile '''**''' et 'kwargs' n'est alors plus un tuple mais un dictionnaire associant nom et valeur des variables à transmettre:
<source lang="py">
<source lang="py">
def f(**kwargs):
def f(**kwargs):
Line 112: Line 117:
print f(c=2)
print f(c=2)
</source>
</source>
Là encore, le nommage rend claire l'appel de la fonction.
Là encore, le nommage rend clair l'appel de la fonction.


On peut mixer tous ces mécanismes de passage d'arguments mais en respectant l'ordre suivant: obligatoires, optionnels, extensible non nommés puis extensible nommés soit:
On peut mixer tous ces mécanismes de passage d'arguments mais en respectant l'ordre suivant: arguments obligatoires, optionnels, extensible non nommés puis extensible nommés soit:
<source lang="py">
<source lang="py">
def g(a,b=2,*args,**kwargs):
def g(a,b=2,*args,**kwargs):
Line 142: Line 147:
-->
-->


L'utilisation des listes extensibles d'arguments requiert une bonne maîtrise de la programmation et n'est pas recommandée en première utilisation. Il est idéal d'avoir un nombre fini d'arguments nommés avec valeurs par défaut dans un premier. Cependant, ce mécanisme permet de transmettre des listes d'arguments de façon efficace et évolutive et s'avère utile dans la construction de bibliothèques de fonctions dépendantes les unes les autres.
L'utilisation des listes extensibles d'arguments requiert une bonne maîtrise de la programmation et n'est pas recommandée en première utilisation. Il est idéal d'avoir un nombre fini d'arguments nommés avec valeurs par défaut dans un premier temps. Cependant, ce mécanisme permet de transmettre des listes d'arguments de façon efficace et évolutive et s'avère utile dans la construction de bibliothèques de fonctions dépendantes les unes les autres.


Exemple: une fonction de scipy:
'''Exemple''': La fonction '''plot()''' de '''matplotlib''' est déclarée avec la 'signature universelle'
<source lang="py">
def matplotlib.pyplot.plot(*args, **kwargs):
</source>
mais il est clair qu'en pratique, on ne pourra pas lui donner n'importe quoi comme arguments. Pour pouvoir comprendre quoi lui passer comme arguments, il faut lire sa [http://matplotlib.org/api/pyplot_api.html#matplotlib.pyplot.plot documentation]. Essentiellement, les arguments non nommés doivent être des couples 'x,y' d'abscisses/ordonnées avec en option un argument donnant le format. Les arguments nommés sont des options et il faut donc clairement utiliser la syntaxe 'option=valeur' dans le passage d'argument pour qu'elle soit prise en compte.
 
Exemples d'utilisation tirés de la documentation:
<source lang="py">
plot(x,y)
plot(x, y, 'bo')
plot(x1, y1, 'g^', x2, y2, 'g-')
plot(x, y, color='green', linestyle='dashed', marker='o', markerfacecolor='blue', markersize=12).
</source>


== Documentation d'une fonction ==
== Documentation d'une fonction ==

Revision as of 09:53, 27 August 2015

<--Sommaire

Définir une fonction

Deux options suivant la complexité

1. définition usuelle à l'aide du mot-clé def et ouverture d'un bloc. return permet que la fonction renvoie un résultat mais ce n'est pas obligatoire, la fonction peut simplement manipuler les données en entrée

def f(x):
    return x**2

2. on peut également définir une fonction lambda ou 'fonction anonyme' selon

f = lambda x: x**2

dans l'exemple ci-dessus, le caractère anonyme n'est pas très clair car on lui a affecté la variable f pour l'utiliser avec f(x) comme précédemment le caractère anonyme apparaît lorsqu'on passe ce type de fonction en argument.

Récursivité

Une fonction peut s'appeler elle-même

def f(x):
    if x < 10: return x*f(x+1)
    else: return 10
print f(1), f(10), f(9), f(2.34)

Fonction à l'intérieur d'une fonction

Il est possible de définir une fonction à l'intérieur d'une autre fonction. Dans ce cas, la fonction n'est visible qu'à l'intérieur de la fonction principale et ne peut être utilisée dans le champ global:

def f(x):
    from math import exp
    def g(y): return y**2+1
    return exp(g(x)) / g(x)
print f(1), g(2)

Dans l'exemple, l'affichage de g(2) renvoie une erreur. Cela peut être très pratique pour structure le corps d'une fonction complexe sans avoir à polluer le niveau global avec beaucoup de fonctions.


Passage d'arguments d'une fonction

Dans ce qui suit, arguments et paramètres seront considérés comme synonymes.

  • f.__module__ : affiche le module de rattachement d'une fonction
  • f.__name__ : affiche le nom de la fonction, par exemple si g = f, g.__name__ donne 'f'

Signature d'une fonction: c'est la combinaison du nom et de la liste d'arguments

def f(a,b,c): 
    return a+b**2+c**3
f(1,2,3)
print f.__module__
g = f
print g.__name__

définit localement les variables a,b,c qui prendront comme valeurs celles transmises lors de l'appel de la fonction, par exemple f(1,2,3). Il est obligatoire de donner la liste des arguments et dans le bon ordre.

Arguments optionnels: il est possible de mettre des arguments optionnels à condition de leur donner une valeur par défaut et ils doivent nécessairement être placés après les arguments obligatoires:

def f(a,b,c=0):  # c est optionnel donc f(1,2) fonctionne et c prendra la valeur 0
    return a+b**2+c**3
print f.__defaults__ # affiche les valeurs par défaut des arguments optionnels
def f(a=1,b=2,c=3):
    return a+b**2+c**3
print f.__defaults__
f(1,2,3), f(2)

Arguments nommés: lorsqu'il y a des paramètres optionnels, on peut ne passer que l'argument dont on veut attribuer une valeur différente de la valeur par défaut, sans avoir à respecter l'ordre:

f(b=3)
f(c=2,a=2)

On parle de paramètres nommés. Cela donne une lecture claire de l'appel à la fonction. Si l'on utilise des arguments non nommés et des arguments nommés, ces derniers sont toujours placés à la fin:

f(1,c=3)

Liste extensible d'arguments non nommés: Python permet à une fonction d'accepter une liste extensible d'arguments, c'est-à-dire que le nombre d'arguments n'est pas défini à l'avance. Il faudra dans ce cas veiller à ce que la fonction gère correctement cette liste quelconque d'arguments. La syntaxe est la suivante, avec une * devant le nom de ce qui sera un tuple contenant la liste des paramètres (on aime bien utiliser 'args' mais on peut appeler ce tuple autrement):

def f(*args):
    print args
    if len(args) == 3:
        a,b,c = args[0],args[1],args[2]
        return a+b+c
    else:
        return None
f(1,2,3)
f(3)

Liste extensible d'arguments nommés: il est en fait plus judicieux d'utiliser une liste de paramètres nommés extensible. La syntaxe est la suivante, avec cette fois une double étoile ** et 'kwargs' n'est alors plus un tuple mais un dictionnaire associant nom et valeur des variables à transmettre:

def f(**kwargs):
    print str(kwargs)
    if len(kwargs) == 3:
        a,b,c = kwargs['a'],kwargs['b'],kwargs['c']
        return a+b**2+c**3
    else:
        return None
print f(a=1,b=2,c=3),f(c=1,b=2,a=3)
print f(c=2)

Là encore, le nommage rend clair l'appel de la fonction.

On peut mixer tous ces mécanismes de passage d'arguments mais en respectant l'ordre suivant: arguments obligatoires, optionnels, extensible non nommés puis extensible nommés soit:

def g(a,b=2,*args,**kwargs):
    print a,b,args,kwargs
g(1,2,3,4,5,x=3)


L'utilisation des listes extensibles d'arguments requiert une bonne maîtrise de la programmation et n'est pas recommandée en première utilisation. Il est idéal d'avoir un nombre fini d'arguments nommés avec valeurs par défaut dans un premier temps. Cependant, ce mécanisme permet de transmettre des listes d'arguments de façon efficace et évolutive et s'avère utile dans la construction de bibliothèques de fonctions dépendantes les unes les autres.

Exemple: La fonction plot() de matplotlib est déclarée avec la 'signature universelle'

def matplotlib.pyplot.plot(*args, **kwargs):

mais il est clair qu'en pratique, on ne pourra pas lui donner n'importe quoi comme arguments. Pour pouvoir comprendre quoi lui passer comme arguments, il faut lire sa documentation. Essentiellement, les arguments non nommés doivent être des couples 'x,y' d'abscisses/ordonnées avec en option un argument donnant le format. Les arguments nommés sont des options et il faut donc clairement utiliser la syntaxe 'option=valeur' dans le passage d'argument pour qu'elle soit prise en compte.

Exemples d'utilisation tirés de la documentation:

plot(x,y)
plot(x, y, 'bo')
plot(x1, y1, 'g^', x2, y2, 'g-')
plot(x, y, color='green', linestyle='dashed', marker='o', markerfacecolor='blue', markersize=12).

Documentation d'une fonction

On peut écrire aisément une documentation pour une fonction en écrivant sous la définition un texte insérer dans des triples guillemets. On peut alors faire afficher la documentation à l'aide de la fonction help ou en faisant afficher l'attribut __doc__ de la fonction qui conserve le texte.

def Sort(L, cmp):
    """
    Fonction Sort(L): 
    prend une liste L en argument et la tri selon la méthode cmp.
    """
help(Sort)
print Sort.__doc__

Appliquer une fonction à une liste

une suppose une fonction f() définie, par exemple

f = lambda x: x**2
A = range(10)# initialisation d'une liste

une première version lourde mais qui ressemble au C

B = [ 0.0 ]*len(A)
for i in range(len(A)):
    B[i] = f(A[i])

une version similaire mais écrite en ligne

B = [ f(A[i]) for i in range(len(A)) ]

le même résultat en itérant directement sur les éléments de A

B = [ f(a) for a in A ]

en utilisant la fonction map du standard Python

B = map(f,A)

enfin le tout en une ligne, possible uniquement avec une définition faite par le mot-clé lambda

B = map(lambda x: x**2,A)