【问题标题】:Python Child Class with multiple init methods?具有多个init方法的Python子类?
【发布时间】:2017-07-25 01:13:58
【问题描述】:

我有一个我在 Python2.7 中构建的应用程序,它可以按原样工作,但感觉不干净,发生的事情也不是很明确,所以如果我暂时离开代码,我有很难记住它实际上是如何在引擎盖下工作的,这显然不好。我已经重构了代码,它看起来更明确,但实际上并不是更清晰。

我试图找出以两种不同方式初始化这些类的最简洁方法 - 1)从用户生成的实例化(如果在程序执行期间从头开始添加新对象),或 2)从从 JSON 导入对象的历史记录(来自先前的程序执行)。这是我处理此问题的最新方法:

class Device(object):
    def __init__(self, dev_type, preset_prefix, default_preset,
                 from_json=False, json_path=None, **device_attrs):

    if not from_json:  # otherwise set in child class __init__
        self.name = device_attrs['name']
        self.sn = device_attrs['sn']
        self.mfg = device_attrs['mfg']
        self.tech = device_attrs['tech']
        self.model = device_attrs['model']
        self.sw_ver = device_attrs['sw_ver']
        self.hours = 0
    else:
        self.hours = device_attrs['hours']

    self.type = dev_type
    self.json = json_path
    self.preset_prefix = preset_prefix
    self.preset = default_preset

class Monitor(Device):
    def __init__(self, name, sn, mfg, tech, model, sw_ver, from_json=False,
                 json_path=None, **monitor_dict):
        if from_json:
            self.__dict__ = monitor_dict
        device_properties = {'name': name, 'sn': sn, 'mfg': mfg, 'tech': tech,
                             'model': model, 'sw_ver': sw_ver}
        monitor_dict.update(device_properties)
        super(Monitor, self).__init__('monitor', 'user', 1, from_json,
                                      json_path, **monitor_dict)
        if cals:
            self._init_cal_from_json(monitor_dict['cals'])

现在我可以从以前保存的 JSON 进行初始化(从这个对象生成,所以我可以确定键/值对是正确的):

my_monitor = Monitor(from_json=True, json_path=device_json_file, **device_json_dict))

或者作为一个从头开始的新对象:

my_monitor = Monitor('monitor01', '12345', 'HP', 'LCD',
                     'HP-27', 'v1.0')

这似乎有点混乱,但仍然比我的原始版本更好,它没有任何位置参数用于子 init(使得很难知道必须传入哪些数据), **monitor_dict 希望它包含正确的键/值对。然而,这种将这些参数合并到字典中的方法似乎很奇怪,但我已经多次重构了这个方法,这似乎是最干净的方法。

这是以多种方式处理初始化对象的最佳方式,还是我可以以某种方式创建两个单独的 init 函数,一个用于从 JSON 加载,一个用于新创建全新的对象?

【问题讨论】:

  • 具有额外的构造方法(使用@classmethod 装饰器),如answer below 所示,是一个非常典型的Python 习惯用法。标准库中有许多示例(想到的一个是dict.fromkeys() 方法)。将__init__ 用于最常用或最期望的构造函数,并根据需要创建其他构造函数。所以我建议添加一个from_json 方法。
  • 嗨@ice,我已经检查了编辑过的问题,如果你可以提出建议,你可能应该问另一个问题,对于新问题,我的意思是,我的回答只解决你的主要和原始问题(对于python 3,我的错误),如果你提出一个新问题,新问题(以及这个问题的链接)你会成功,我可以保证它发生在我帮助的其他人之前。如果您提出新问题,请告诉我所有详细信息:),我希望我能提供更多帮助

标签: python json python-2.7 oop subclass


【解决方案1】:

我更喜欢创建新的构造函数作为类方法,像这样,你可以根据需要创建更多,或者根据需要进行调整:

class Device(object):
    def __init__(self, dev_type, preset_prefix, default_preset, json_path=None, **device_attrs):


      self.name = device_attrs['name']
      self.sn = device_attrs['sn']
      self.mfg = device_attrs['mfg']
      self.tech = device_attrs['tech']
      self.model = device_attrs['model']
      self.sw_ver = device_attrs['sw_ver']
      self.hours = 0

      self.type = dev_type
      self.json = json_path
      self.preset_prefix = preset_prefix
      self.preset = default_preset

class Monitor(Device):


    @classmethod
    def new_from_json(self, name, sn, mfg, tech, model, sw_ver, json_path=None, **monitor_dict):
        self.__dict__ = monitor_dict
        device_properties = {'name': name, 'sn': sn, 'mfg': mfg, 'tech': tech,
                             'model': model, 'sw_ver': sw_ver}
        monitor_dict.update(device_properties)
        super(Monitor, self).__init__('monitor', 'user', 1,
                                      json_path, **monitor_dict)

举个例子:

class Parent():
    def __init__(self,some):
        self.some = some


class Object(Parent):

    @classmethod
    def new_from_dict(self,some):
        Parent.__init__(self,some)
        self.adress = {"Me": 123}
        return self

然后:

obj = Object.new_from_dict("ME")
obj.adress

{"Me": 123}

obj.some

"ME"

【讨论】:

  • 谢谢,我以前读过关于类方法和工厂的文章,但并没有真正将两个和两个放在一起来说明如何在这里应用它。在您在这里回答之后进行一些额外的阅读让我得到了我认为是一个干净、合乎逻辑的解决方案。我已经更新了我的问题以包含修订版。
  • 我实际上是在尝试更接近您在此处为不同类型的课程所做的事情,而我的修改后的代码并没有真正适用或工作。我遇到的问题是我无法理解的 - 在您提供的代码中,new_from_json() 中的 self 指的是什么?在代码中的那一点上,没有任何东西被初始化,所以 self 没有引用任何特定的对象,并且 self.__dict__ 不会实际工作?我错过了什么吗?
  • @iceblueorbitz 我会在最后为你添加一个例子,所以也许你能看得更清楚
  • 这样做仍然会导致同样的问题:TypeError: unbound method __init__() must be called with Parent instance as first argument (got type instance instead)。当使用 classmethod 装饰器时,PyCharm 建议使用 cls 代替 self,然后使用 cls(some) 进行初始化,但这将调用子类“Object”init,在这种情况下不会存在。
  • @iceblueorbitz 尝试更改 super 的类,我在 python 3.6 中完成了它并且工作正常。这个错误很奇怪..
猜你喜欢
  • 2021-05-03
  • 1970-01-01
  • 2019-11-25
  • 1970-01-01
  • 1970-01-01
  • 2011-05-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多