web-dev-qa-db-fra.com

Python: classe abstraite __init__

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?

20
dabadaba

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__
25
Mike Müller

Vous devez également définir les méthodes comme abstraites avec le @abc.abstractmethod décorateur.

3

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
3
Ashwini Chaudhary

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.
2
gipsy