Lier dynamiquement les méthodes non liées
En Python, nous rencontrons souvent des situations où nous devons lier une méthode non liée à une instance sans l'invoquer. Cela peut s'avérer une technique précieuse dans divers scénarios, tels que la création d'interfaces graphiques dynamiques ou la gestion d'événements de manière structurée.
Le problème de l'explosion des programmes
Considérez le code suivant extrait :
<code class="python">class MyWidget(wx.Window): buttons = [ ("OK", OnOK), ("Cancel", OnCancel) ] def setup(self): for text, handler in MyWidget.buttons: b = wx.Button(parent, label=text).bind(wx.EVT_BUTTON, handler)</code>
Le problème ici est que le gestionnaire représente des méthodes non liées, ce qui provoque le crash du programme avec une erreur. Pour résoudre ce problème, nous avons besoin d'un moyen de lier ces méthodes indépendantes à l'instance spécifique de MyWidget.
La puissance des descripteurs
Les méthodes de Python sont également des descripteurs, qui fournissent un moyen de les lier dynamiquement. En appelant la méthode spéciale __get__ sur la méthode non liée, nous pouvons obtenir une méthode liée :
<code class="python">bound_handler = handler.__get__(self, MyWidget)</code>
En attribuant la méthode liée à un attribut de niveau classe, nous pouvons la lier efficacement à l'instance :
<code class="python">setattr(self, handler.__name__, bound_handler)</code>
Une fonction de liaison réutilisable
En utilisant cette technique, nous pouvons créer une fonction réutilisable pour lier des méthodes non liées :
<code class="python">def bind(instance, func, as_name=None): """ Bind the function *func* to *instance*, with either provided name *as_name* or the existing name of *func*. The provided *func* should accept the instance as the first argument, i.e. "self". """ if as_name is None: as_name = func.__name__ bound_method = func.__get__(instance, instance.__class__) setattr(instance, as_name, bound_method) return bound_method</code>
Avec cette fonction , nous pouvons maintenant lier les méthodes non liées comme suit :
<code class="python">bind(something, double) something.double() # returns 42</code>
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!