Considérez cette classe:
class foo(object):
pass
La représentation de chaîne par défaut ressemble à ceci:
>>> str(foo)
"<class '__main__.foo'>"
Comment puis-je faire de cet affichage une chaîne personnalisée?
Considérez cette classe:
class foo(object):
pass
La représentation de chaîne par défaut ressemble à ceci:
>>> str(foo)
"<class '__main__.foo'>"
Comment puis-je faire de cet affichage une chaîne personnalisée?
Réponses:
Implémentez __str__()
ou __repr__()
dans la métaclasse de la classe.
class MC(type):
def __repr__(self):
return 'Wahaha!'
class C(object):
__metaclass__ = MC
print C
À utiliser __str__
si vous entendez une chaîne de caractères lisible, à utiliser __repr__
pour des représentations non ambiguës.
_representation
au corps de classe et return self._representation
dans la __repr__()
méthode de la métaclasse.
__repr__
pour représenter C
. Une alternative à avoir un _representation
membre est de créer une fabrique de métaclasses qui produit une métaclasse avec le bon __repr__
(cela pourrait être bien si vous l'utilisez beaucoup).
class foo(object):
def __str__(self):
return "representation"
def __unicode__(self):
return u"representation"
instances
la classe, pas pour la classe elle-même.
Si vous devez choisir entre __repr__
ou __str__
opter pour le premier, comme par défaut l'implémentation __str__
appelle __repr__
quand elle n'a pas été définie.
Exemple de Vector3 personnalisé:
class Vector3(object):
def __init__(self, args):
self.x = args[0]
self.y = args[1]
self.z = args[2]
def __repr__(self):
return "Vector3([{0},{1},{2}])".format(self.x, self.y, self.z)
def __str__(self):
return "x: {0}, y: {1}, z: {2}".format(self.x, self.y, self.z)
Dans cet exemple, repr
renvoie à nouveau une chaîne qui peut être directement consommée / exécutée, alors qu'elle str
est plus utile comme sortie de débogage.
v = Vector3([1,2,3])
print repr(v) #Vector3([1,2,3])
print str(v) #x:1, y:2, z:3
__repr__
vs __str__
soit correct, cela ne répond pas à la question réelle, qui concerne les objets de classe, pas les instances.
La réponse approuvée d'Ignacio Vazquez-Abrams est tout à fait juste. Il s'agit cependant de la génération Python 2. Une mise à jour pour le Python 3 actuel serait:
class MC(type):
def __repr__(self):
return 'Wahaha!'
class C(object, metaclass=MC):
pass
print(C)
Si vous voulez du code qui s'exécute à la fois sur Python 2 et Python 3, le module six vous a couvert:
from __future__ import print_function
from six import with_metaclass
class MC(type):
def __repr__(self):
return 'Wahaha!'
class C(with_metaclass(MC)):
pass
print(C)
Enfin, si vous avez une classe que vous souhaitez avoir une représentation statique personnalisée, l'approche basée sur les classes ci-dessus fonctionne très bien. Mais si vous en avez plusieurs, vous devrez générer une métaclasse similaire à MC
chacune, et cela peut devenir fastidieux. Dans ce cas, aller plus loin dans votre métaprogrammation et créer une fabrique de métaclasses rend les choses un peu plus propres:
from __future__ import print_function
from six import with_metaclass
def custom_class_repr(name):
"""
Factory that returns custom metaclass with a class ``__repr__`` that
returns ``name``.
"""
return type('whatever', (type,), {'__repr__': lambda self: name})
class C(with_metaclass(custom_class_repr('Wahaha!'))): pass
class D(with_metaclass(custom_class_repr('Booyah!'))): pass
class E(with_metaclass(custom_class_repr('Gotcha!'))): pass
print(C, D, E)
impressions:
Wahaha! Booyah! Gotcha!
La métaprogrammation n'est pas quelque chose dont vous avez généralement besoin tous les jours, mais quand vous en avez besoin, elle frappe vraiment!
Ajoutant à toutes les bonnes réponses, ma version avec décoration:
from __future__ import print_function
import six
def classrep(rep):
def decorate(cls):
class RepMetaclass(type):
def __repr__(self):
return rep
class Decorated(six.with_metaclass(RepMetaclass, cls)):
pass
return Decorated
return decorate
@classrep("Wahaha!")
class C(object):
pass
print(C)
stdout:
Wahaha!
Les inconvénients:
C
sans super classe (non class C:
)C
les instances seront des instances d'une dérivation étrange, c'est donc probablement une bonne idée d'ajouter également un __repr__
pour les instances.