@abstractmethod 由要继承的第二个类给出(使用抽象方法的多重继承)

Posted

技术标签:

【中文标题】@abstractmethod 由要继承的第二个类给出(使用抽象方法的多重继承)【英文标题】:@abstractmethod given by a second class to inherit from (multiple inheritance with abstract methods) 【发布时间】:2022-01-08 23:09:02 【问题描述】:

我想定义一个抽象方法,从我的基类通过第二次继承。

由于我可以想象我的问题甚至这句话令人困惑,这里是我正在尝试做的一个例子:

想象以下 BaseClass 包含 2 个抽象方法:处理和验证

class BaseClass(ABC):
"""
Base class for all services
"""

def __init__(self) -> None:
    pass

@abstractmethod
def validate_input(self, payload: dict) -> None:
    """
    validates the payload
    """
    pass

@abstractmethod
def process(self, payload: dict) -> dict:
    """
    processes the payload
    """
    pass

如您所知,BaseClass 中有 2 个方法需要由进程定义。我现在想在继承自 BaseClass 的 Process1 类中定义流程方法。但是我还需要定义 validate_input 方法,对于不同的进程可以是相同的。因此我想到了使用多重继承来解决这个问题。 就我而言,我想创建第二个 BaseValidation1 类,其中包含某种验证方法,例如检查有效负载中是否有密钥。

class Validation1(ABC):
"""
Validates if the payload is valid given a certained method e.g. if payload contains a certain key
"""
def validate_input(self, payload_dict: typing.Dict[str, typing.Any]) -> None:
    """
    check if dict contains a certain key
    """
    if not payload_dict:
        raise ValidationException("payload is empty")
    if not payload_dict.get("key"):
        raise ValidationException("payload does not contain key")

所以现在我想从两者继承来定义我的 Process1。

class Process1(BaseClass,Validation1):
"""
Base class for processing the payload
"""
def process(self, payload: typing.Dict) -> typing.Dict:
    """
    Process the payload
    """
    processed_payload = payload
    return processed_payload
    
  

但是,我非常不确定我的方法是否是解决此问题的最佳方法。此外,Pylint 已经显示以下警告:

E0110: Abstract class 'Process1' with abstract methods instantiated (abstract-class-instantiated)

欣赏任何解决方案。

【问题讨论】:

【参考方案1】:

如果您颠倒基类的顺序,则您的示例有效:

class Process1(Validation1, BaseClass):
    ...

原因与python's method resolution order有关:您定义Process1的父类的方式导致python解释器在您调用Process1.validate()时“找到”BaseClass.validate(),这是抽象的。

我将进一步从Validation1 中删除ABC 父级,因为我认为该类是Mixin。这是一个完全可用的最小示例:

from abc import ABC, abstractmethod


class BaseClass(ABC):

    def __init__(self):
        pass

    @abstractmethod
    def validate(self):
        pass

    @abstractmethod
    def process(self):
        pass


class ValidationMixin:

    def validate(self):
        pass


class Processor(ValidationMixin, BaseClass):

    def process(self):
        pass

【讨论】:

以上是关于@abstractmethod 由要继承的第二个类给出(使用抽象方法的多重继承)的主要内容,如果未能解决你的问题,请参考以下文章

重载虚函数集的部分继承

js实现继承的几种方式

命名查询和继承

正确包含标题

更改不同类中的变量的值,而无需重写方法

如何在 ios 应用程序中以编程方式移动到第二个视图?