And*_*ing 4 python abc abstract-base-class
据我所知,Python 模块 abc 应该防止类的实例化,这些类没有@abstractmethod
实现基类的所有标记方法(前提是基类已经__metaclass__ = ABCMeta
设置)
但是,这似乎不适用于以下代码:
抽象基类:
""" Contains payment processors for executing payments """
from abc import ABCMeta, abstractmethod
class AbstractPaymentProcessor:
""" Abstract class for executing faucet Payments
Implement this at your own. Possible implementations include
online wallets and RPC calls to running dogecoin wallets """
__metaclass__ = ABCMeta
@abstractmethod
def execute_payment(self, destination_address, amount):
""" Execute a payment to one receiving single address
return the transaction id or None """
pass
@abstractmethod
def execute_multi_payment(self, destination_addresses, amounts):
""" Execute a payment to multiple receiving addresses
return the transaction id or None """
pass
@abstractmethod
def get_transaction_status(self):
""" Get the status of the transaction
Indicate if transaction is already confirmed. Return
- True if confirmed
- False if unconfirmed
- None if transaction doesn't exist (or raise exception?)"""
pass
@abstractmethod
def get_available_balance(self):
""" Get the available balance
i.e. how much "cash" is in the faucet """
pass
Run Code Online (Sandbox Code Playgroud)
子类缺少一个方法:
""" Contains a logging payment processor """
import logging
import random
from AbstractPaymentProcessor import AbstractPaymentProcessor
class DummyLoggingPaymentProcessor (AbstractPaymentProcessor):
""" Payment processor that does nothing, just logs """
def __new__(self):
self._logger = logging.getLogger(__name__)
self._logger.setLevel(logging.INFO)
def execute_payment(self, destination_address, amount):
""" Execute a payment to one receiving single address
return the transaction id or None """
raise NotImplementedError("Not implemented yet")
def execute_multi_payment(self, destination_addresses, amounts):
""" Execute a payment to multiple receiving addresses
return the transaction id or None """
raise NotImplementedError("Not implemented yet")
def get_transaction_status(self):
""" Get the status of the transaction
Indicate if transaction is already confirmed. Return
- True if confirmed
- False if unconfirmed
- None if transaction doesn't exist """
raise NotImplementedError("Not implemented yet")
if __name__ == '__main__':
# can instanciate, although get_available_balance is not defined. Why? abc should prevent this!?
c = DummyLoggingPaymentProcessor()
c.get_available_balance()
Run Code Online (Sandbox Code Playgroud)
子类可以在(相当粗糙的)测试代码中实例化。为什么呢?
我正在使用 Python 2.7。
你是压倒性的__new__
;正是这种方法 (on object.__new__
) 阻止了实例化。
您没有在此处创建不可变类型或以其他方式更改新对象的创建,因此请__init__
改用:
def __init__(self):
self._logger = logging.getLogger(__name__)
self._logger.setLevel(logging.INFO)
Run Code Online (Sandbox Code Playgroud)
__new__
无论如何你都用错了;传入的第一个参数是class,而不是实例,因为此时尚未创建任何实例。通过覆盖__new__
而不是调用原始实例,您 a) 不会创建实例,并且 b) 不会触发阻止首先创建实例的代码。
使用__init__
代替__new__
实例化会按预期引发异常:
>>> class DummyLoggingPaymentProcessor (AbstractPaymentProcessor):
... """ Payment processor that does nothing, just logs """
... def __init__(self):
... self._logger = logging.getLogger(__name__)
... self._logger.setLevel(logging.INFO)
... def execute_payment(self, destination_address, amount):
... """ Execute a payment to one receiving single address
...
... return the transaction id or None """
... raise NotImplementedError("Not implemented yet")
... def execute_multi_payment(self, destination_addresses, amounts):
... """ Execute a payment to multiple receiving addresses
...
... return the transaction id or None """
... raise NotImplementedError("Not implemented yet")
... def get_transaction_status(self):
... """ Get the status of the transaction
...
... Indicate if transaction is already confirmed. Return
... - True if confirmed
... - False if unconfirmed
... - None if transaction doesn't exist """
... raise NotImplementedError("Not implemented yet")
...
>>> c = DummyLoggingPaymentProcessor()
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: Can't instantiate abstract class DummyLoggingPaymentProcessor with abstract methods get_available_balance
Run Code Online (Sandbox Code Playgroud)