J'essaie de déclarer une classe abstraite A
avec un constructeur avec un comportement par défaut: toutes les sous-classes doivent initialiser un membre self.n
:
from abc import ABCMeta
class A(object):
__metaclass__ = ABCMeta
def __init__(self, n):
self.n = n
Cependant, je ne veux pas laisser la classe A
instanciée car, eh bien, c'est une classe abstraite. Le problème est que cela est en fait autorisé:
a = A(3)
Cela ne produit aucune erreur, alors que je m'y attendrais.
Alors: comment puis-je définir une classe abstraite non instanciable tout en définissant un comportement par défaut pour le constructeur?
Faire le __init__
une méthode abstraite:
from abc import ABCMeta, abstractmethod
class A(object):
__metaclass__ = ABCMeta
@abstractmethod
def __init__(self, n):
self.n = n
if __name__ == '__main__':
a = A(3)
aide:
TypeError: Can't instantiate abstract class A with abstract methods __init__
Version Python 3:
from abc import ABCMeta, abstractmethod
class A(object, metaclass=ABCMeta):
@abstractmethod
def __init__(self, n):
self.n = n
if __name__ == '__main__':
a = A(3)
Fonctionne également:
TypeError: Can't instantiate abstract class A with abstract methods __init__
Vous devez également définir les méthodes comme abstraites avec le @abc.abstractmethod
décorateur.
Vous pouvez remplacer __new__
méthode pour empêcher l'instanciation directe.
class A(object):
__metaclass__ = ABCMeta
def __new__(cls, *args, **kwargs):
if cls is A:
raise TypeError(
"TypeError: Can't instantiate abstract class {name} directly".format(name=cls.__name__)
)
return object.__new__(cls)
Sortie:
>>> A()
Traceback (most recent call last):
File "<ipython-input-8-3cd318a12eea>", line 1, in <module>
A()
File "/Users/ashwini/py/so.py", line 11, in __new__
"TypeError: Can't instantiate abstract class {name} directly".format(name=cls.__name__)
TypeError: TypeError: Can't instantiate abstract class A directly
Une solution moins élégante peut être la suivante:
class A(object):
def __init__(self, n):
if self.__class__ == A:
raise Exception('I am abstract!')
self.n = n
Usage
class B(A):
pass
a = A(1) # Will throw exception
b = B(1) # Works fine as expected.