поднять флаг, если атрибут изменен приводит к слишком много свойств? - PullRequest
0 голосов
/ 31 октября 2018

Для моего приложения у меня есть объект, который имеет некоторый набор атрибутов. Один набор атрибутов, параметры, могут быть доступны и настроены пользователем. Другой набор атрибутов, выходные данные, должен быть доступен пользователю, но рассчитывается с использованием внутренних методов. Кроме того, если какой-либо из атрибутов параметра настроен, выходные данные также должны быть пересчитаны из внутренних методов и соответствующим образом скорректированы. Однако, поскольку эти вычисления могут быть дорогостоящими, я не хочу выполнять их без необходимости, если (или до) они не запрошены.

В настоящее время я могу реализовать это, сделав каждый атрибут «параметра» свойством и включив флаг self.calculated, который поднимается при изменении любого из параметров, а также делая каждый атрибут «вывода» свойством, которое проверяет self.calculated flag и, соответственно, либо возвращает выходные данные напрямую, если вычисления не требуются, либо выполняет вычисления, понижает флаг и возвращает выходные данные.

См. Код

class Rectangle(object):

    def __init__(self, length=1, width=1):
        self._length = length
        self._width = width
        self._area = self.calc_area()
        self._perim = self.calc_perim()
        self.calculated = True

    @property
    def length(self):
        return self._length

    @length.setter
    def length(self, value):
        if value != self._length:
            self._length = value
            self.calculated = False

    @property
    def width(self):
        return self._width

    @width.setter
    def width(self, value):
        if value != self._width:
            self._width = value
            self.calculated = False

    @property
    def area(self):
        if self.calculated is True:
            return self._area
        else:
            self.recalculate()
            return self._area

    @property
    def perim(self):
        if self.calculated is True:
            return self._perim
        else:
            self.recalculate()
            return self._perim

    def calc_area(self):
        return self.length * self.width

    def calc_perim(self):
        return 2 * (self.length + self.width)

    def recalculate(self):
        self._area = self.calc_area()
        self._perim = self.calc_perim()
        self.calculated = True

    def double_width(self):
        self.width = 2 * self.width

Это дает желаемое поведение, но кажется чрезмерным распространением свойств, что было бы особенно проблематично, если бы было большое количество параметров и выходов.

Есть ли более чистый способ реализации этой структуры изменения / пересчета атрибутов? Я нашел пару постов, где представлено решение, включающее написание метода __setattr__ для класса, но я не уверен, будет ли это просто реализовать в моем, так как поведение должно отличаться в зависимости от конкретного атрибута, являющегося задавать. Я полагаю, что это можно сделать с помощью проверки в методе __setattr__ о том, является ли атрибут параметром или выводом ...

Оформление класса для отслеживания изменений атрибутов

Как определить, когда устанавливается атрибут атрибута?

1 Ответ

0 голосов
/ 31 октября 2018

Существует несколько различных опций, и их использование сильно зависит от варианта использования. Очевидно, что приведенные ниже варианты могут быть довольно плохими во многих случаях использования.

Удаление выходов при изменении входов

Пример реализации этого:

@width.setter
def width(self, value):
    if value != self._width:
        self._width = value
        (self._area,self._perim) = (None,None)

def perim(self):
    if not self._perim:
        self._perim = calc_perim(self)
    return self._perim

Это не решает большинство ваших проблем, но избавляет от флага calculated (и хотя ваш код пересчитывает все выходные данные, когда любой из них запрашивается после обновления, этот код просто вычисляет запрошенный один).

Обновление значений при изменении входных данных

Когда вы увеличиваете ширину на x, периметр увеличивается на 2 * x, а площадь увеличивается на x * длину. В некоторых случаях применение таких формул для обновления значений при изменении входных данных может быть более эффективным, чем вычисление выходных данных с нуля при каждом изменении входных данных.

Отслеживать последние значения

Всякий раз, когда вы рассчитываете выходные данные, следите не только за тем, что вы получили, но и за тем, какие данные вы использовали для их расчета. Затем в следующий раз, когда у объекта спрашивают, каковы его выходные данные, он может проверить, были ли выходные данные рассчитаны в соответствии с его текущими атрибутами. Очевидно, что это требует умножения входного пространства хранения.

Памятка

Пойдя даже дальше, чем предыдущий параметр, создайте словарь, в котором ключи являются кортежами атрибутов, а значения выводятся. Если у вас есть функция calculate_output(attributes), замените все вызовы этой функции на

def output_lookup(attributes):
   if not attributes in output_dict.keys():
          output_dict[attributes] = calculate_output(attributes)
   return output_dict[attributes]

Вам следует использовать эту опцию, если вы ожидаете, что определенные комбинации атрибутов будут часто повторяться, вычисление выходных данных является дорогостоящим и / или память является дешевой. Это может быть общим для всего класса, поэтому, если у вас есть несколько экземпляров прямоугольников, имеющих одинаковую длину и ширину, вы можете сохранить одно значение (_perim,_area) вместо дублирования его для каждого экземпляра. Так что для некоторых вариантов использования это может быть более эффективным.

Обратите внимание, что ваша проблема в конечном итоге связана с тем фактом, что вы пытаетесь задействовать некоторое запоминание (вы хотите сохранить результаты из своих вычислений, чтобы при обращении к выходным данным объекта вы не Вы должны рассчитать выходные данные, если они уже были рассчитаны для текущих входных данных), но вам необходимо отслеживать, когда, так сказать, «сделать кэш недействительным». Если бы вы просто рассматривали область и периметр как методы, а не атрибуты, или вы должны были рассматривать экземпляры как неизменяемые и требовать сброса атрибутов путем создания нового экземпляра с новыми значениями, вы бы исключили сложность, которую вы добавили на длину и ширину. Вы не можете иметь все это: вы не можете иметь кэшированные значения из изменяемых атрибутов без некоторых накладных расходов.

PS is True является избыточным в if self.calculated is True:.

...