Как передать экземпляр класса в качестве параметра декоратору, украшающему метод экземпляра, определенный в классе? - PullRequest
0 голосов
/ 26 декабря 2018

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

Это то, что я пытался сделать (пожалуйста, проверьте реализацию методов push и pop):

repl.it link: https://repl.it/@Ishitva/Stack

class StackFullException(Exception):
    pass

class StackEmptyException(Exception):
    pass

def checkStackFull(instance):
    def check(func):
        def execute(*args, **kwargs):
            if len(instance.items) <= instance.limit:
                return func(*args, **kwargs)
            raise StackFullException

        return execute
    return check

def checkStackEmpty(instance):
    def check(func):
        def execute(*args, **kwargs):
            if len(instance.items) > -1:
                return func(*args, **kwargs)
            raise StackEmptyException

        return execute
    return check


class Stack():

    def __init__(self, limit=10):
        self.items = []
        self.limit = limit

    @checkStackFull(self)
    def push(item):
        self.items.append(item)
        return item

    @checkStackEmpty(self)
    def pop():
        return self.items.pop()

    def getSize():
        return len(self.items)

Это даетмне следующее исключение:

Traceback (most recent call last):
  File "main.py", line 28, in <module>
    class Stack():
  File "main.py", line 34, in Stack
    @checkStackFull(self)
NameError: name 'self' is not defined

1 Ответ

0 голосов
/ 26 декабря 2018

Но если вам действительно нужно это сделать, тогда код:

class StackFullException(Exception):
    pass

class StackEmptyException(Exception):
    pass


def checkStackFull(func):
    def execute(self, *args, **kwargs):
        if len(self.items) <= self.limit:
            return func(self, *args, **kwargs)
        raise StackFullException()

    return execute


def checkStackEmpty(func):
    def execute(self, *args, **kwargs):
        if len(self.items):
            return func(self, *args, **kwargs)
        raise StackEmptyException()
    return execute


class Stack():
    def __init__(self, limit=10):
        self.items = []
        self.limit = limit

    @checkStackFull
    def push(self, item):
        self.items.append(item)
        return item

    @checkStackEmpty
    def pop(self):
        return self.items.pop()

    def getSize(self):
        return len(self.items)

И, кстати, всплывающее окно из пустого списка в любом случае вызовет IndexError, так что вы можете просто использовать это.

...