能够实例化python类,尽管它是抽象的(使用abc)

时间:2014-06-15 23:48:09

标签: python python-3.x abstract-class python-2.x abstract-base-class

这是参考this question对“使用python的abc模块创建抽象类”的答案。 (由@alexvassel作为答案接受)。

我尝试了这些建议,但奇怪的是,尽管遵循了使用abc方式的建议,但它对我不起作用。因此我在这里将其作为一个问题发布:

这是我的Python代码:

from abc import ABCMeta, abstractmethod

class Abstract(object):
    __metaclass__ = ABCMeta

    @abstractmethod
    def foo(self):
        print("tst")

a = Abstract()
a.foo()

当我执行这个模块时,这是我控制台上的输出:

pydev debugger: starting (pid: 20388)
tst

与已接受的答案相反

>>> TypeError: Can not instantiate abstract class Abstract with abstract methods foo

那我在做什么是对还是错?为什么工作而不是失败?感谢任何专家对此的见解。

2 个答案:

答案 0 :(得分:7)

在Python 3中,在创建抽象基类时使用metaclass参数:

from abc import ABCMeta, abstractmethod

class Abstract(metaclass=ABCMeta):

    @abstractmethod
    def foo(self):
        print("tst")

a = Abstract()
a.foo()

答案 1 :(得分:4)

在Python 2中,您必须这样分配元类:

import abc

class ABC(object):

    __metaclass__ = abc.ABCMeta

    @abc.abstractmethod
    def foo(self):
        return True

a = ABC()

提出TypeError

Traceback (most recent call last):
  File "<pyshell#59>", line 1, in <module>
    a = ABC()
TypeError: Can't instantiate abstract class ABC with abstract methods foo

但是在Python 3中,将__metaclass__作为属性进行分配并不起作用(正如您所希望的那样,但是解释器并不认为它是一个错误,只是一个普通的属性,与其他任何属性一样,这就是为什么上面的代码不会引起错误)。现在,元类被定义为类的命名参数:

import abc

class ABC(metaclass=abc.ABCMeta):

    @abc.abstractmethod
    def foo(self):
        return True

a = ABC()

提出TypeError

Traceback (most recent call last):
  File "main.py", line 11, in 
    a = ABC()
TypeError: Can't instantiate abstract class ABC with abstract methods foo