Рекурсивная ошибка при введении setattr в python - PullRequest
3 голосов
/ 26 ноября 2010

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

Кажется, это работает, если я не включаю метод __setattr__. Я могу позвонить «settings.top_travel» и получить ответ обратно. Однако, как только я пытаюсь поставить __setattr__, я, кажется, получаю ошибку.

Это выглядит довольно рекурсивно, поэтому я предполагаю, что Get вызывает Set и т. Д. В части набора атрибутов я хочу, чтобы он записывал обратно в файл конфигурации. Таким образом, всякий раз, когда изменяется один из атрибутов настроек, он сохраняется в файле, из которого он получен.

Ниже вы найдете код и ошибку.

import ConfigParser

class settingsFile(object):

    def __init__(self):

        """
        Reloads the configuration file and returns a dictionary with the 
        settings :
        [config]
        top_travel = 250
        """
        # Create a configuration object and read in the file
        configuration = ConfigParser.ConfigParser()
        configuration.read('config/config.cfg')

        # Return all the "config" section as a list and convert to a dictionary
        self.configuration = dict(configuration.items("config"))

    def refresh(self):

        self.__init__()

    def __getattr__(self, attr):
        return self.configuration[attr]

    def __setattr__(self, attr, value):
        print attr, " is now ", value
        # Do some clever storing with ConfigParser

if __name__ == "__main__":

    settings = settingsFile()
    print settings.top_travel
    settings.top_travel = 600
    print settings.top_travel

Ошибка:

Traceback (most recent call last):
  File "/home/stuff/Documents/Software/Python/dBControllers v2/dBControllers.py", line 52, in <module>
    settings = settingsFile()
  File "/home/stuff/Documents/Software/Python/dBControllers v2/dBControllers.py", line 37, in __init__
    self.configuration = dict(configuration.items("config"))
  File "/home/stuff/Documents/Software/Python/dBControllers v2/dBControllers.py", line 47, in __setattr__
    print self.configuration[attr], " is now ", value
  File "/home/stuff/Documents/Software/Python/dBControllers v2/dBControllers.py", line 44, in __getattr__
    return self.configuration[attr]
  File "/home/stuff/Documents/Software/Python/dBControllers v2/dBControllers.py", line 44, in __getattr__
    return self.configuration[attr]
......
RuntimeError: maximum recursion depth exceeded

Ответы [ 3 ]

5 голосов
/ 26 ноября 2010

Проблема в том, что настройка self.configuration вызывает self.__setattr__

Это можно обойти, изменив назначение для вызова на __setattr__ суперкласса:

class settingsFile(object):

    def __init__(self):
        ...
        # Return all the "config" section as a list and convert to a dictionary
        object.__setattr__(self, 'configuration', dict(configuration.items("config")))
5 голосов
/ 26 ноября 2010

Сделайте __setattr__ эксклюзивным для атрибутов, не начинающихся с '_', и сохраните конфигурацию в self._configuration, затем добавьте требование, чтобы файлы конфигурации не принимали параметры, имена которых начинаются с подчеркивания.

def __setattr__(self, attribute, value):
     if attribute.startswith('_'):
          super(settingsFile, self).__setattr__(attribute, value)
          return
     # Clever stuff happens here
0 голосов
/ 26 ноября 2010

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

...