Разрешено ли использовать определенный пользователем декоратор внутри абстрактного класса? Или его следует использовать после наследования? - PullRequest
1 голос
/ 18 июня 2019

Например, скажем, у меня есть определенный декоратор, который называется: decorate

def decorate(func):
  def inside_func(*args, **kwargs):
    # Do something
    return func(*args, **kwargs)
  return inside_func

Далее, скажем, я пишу абстрактный класс с именем Model

from abc import ABC, abstractmethod

class Model(ABC):
  def __init__(self, value):
      self.value = value
      super().__init__()

  @abstractmethod
  @decorate # <-------------------- IS @decorate ALLOWED HERE?
  def do_something(self):
      pass

ИлиДолжно ли это быть:

from abc import ABC, abstractmethod

class Model(ABC):
  def __init__(self, value):
    self.value = value
    super().__init__()

  @abstractmethod
  def do_something(self):
    pass

# Inherit the base class
class DoAdd42(Model):
   @decorate # <----------------------- SHOULD @decorate BE HERE INSTEAD?
   def do_something(self):
     return self.value + 42

Если разрешены оба варианта, есть ли способ «наилучшей практики» для этого?

1 Ответ

2 голосов
/ 18 июня 2019

Оба разрешены. Потому что abstractmethod, ваши decorate и something в классе - это все функции. Вы можете распечатать их на своей консоли для проверки.

In [2]: def decorate(func):
   ...:   def inside_func(*args, **kwargs):
   ...:     # Do something
   ...:     return func(*args, **kwargs)
   ...:   return inside_func
   ...:
In [7]: from abc import ABC, abstractmethod
   ...:
   ...: class Model(ABC):
   ...:   def __init__(self, value):
   ...:       self.value = value
   ...:       super().__init__()
   ...:
   ...:   @abstractmethod
   ...:   @decorate # <-------------------- IS @decorate ALLOWED HERE?
   ...:   def do_something(self):
   ...:       pass
   ...:   # even no self is valid for class in Python3+
   ...:   def whatever():
   ...:       print('fff')
   ...:

In [8]: print(abstractmethod)
<function abstractmethod at 0x100ff86a8>

In [9]: print(Model.do_something)
<function decorate.<locals>.inside_func at 0x1041031e0>

In [10]: print(Model.whatever)
<function Model.whatever at 0x103b48950>

In [11]: Model.whatever()
fff

Я не знаком с abstractmethod, поэтому я не знаю лучших практик и подробностей о правилах этого. Вы можете попытаться составить собственное суждение, основываясь на информации, которую я вам предоставляю, в сочетании с вашим собственным пониманием abstractmethod.

...