当前位置: 首页 > ai >正文

Python Cookbook-6.18 用__init__参数自动初始化实例变量

任务

想避免编写和维护一种烦人的几乎什么也不做的__init__方法,这种方法中含有一大堆形如 self.something = something 的赋值语句。

解决方案

可以把那些属性赋值任务抽取出来置入一个辅助函数中:

def attributesFromDict(d):self = d.pop('self')for n,y in d.iteritems():setattr(self,n,v)

而__init__方法里的那种千篇一律的赋值语句大概是这个样子的:

def __init__(self,foo,bar,baz,boom=1,bang=2):self.foo = fooself.bar = barself.baz = bazself.boom = boomself.bang = bang

现在可以被缩减为清晰的一行:

def __init__(self,foo,bar,baz,boom=1,bang=2):attributesFromDict(locals())

讨论

如果__init__的主体中没有其他的逻辑,调用内建函数locals 返回的 dict 只包含了被传递给__init__的参数(包括那些并未被传递但却有默认值的参数)。首先函数attributesFromDict 获取对象,假定这个对象是一个名字为“self”的参数,然后将其他的所有元素作为它的属性名进行设置。一个相似但更简单的技术是,不使用辅助函数而是像下面这样:

def __init__(self,foo,bar,baz,boom=1,bang=2):self.__dict__.update(locals())del self.self

不过,后来给出的这种技术同解决方案给出的方法相比,有一个重大的缺陷:它直接设置 self.__dict__中的属性(通过update方法),对于一些特性(property)和高级描述符(descriptor),它无法正常地工作。而解决方案给出的方法使用了内建的 setattr,在这方面表现得很完美。

attributesFromDict 并不适用于使用了更多代码尤其是使用了本地变量的__init__ 方法因为对于传递给它的唯一的字典参数,attributesFromDict无法区分字典里的传递给__init__的参数以及__init__内部的局部变量。如果你在辅助函数中加入一点内省的能力,这个限制就可以被打破:

def attributesFromArguments(d):self = d.pop('self')codeObject = self.__init__.im_func.func_codeargumentNames = codeObject.co_varnames[1:codeObject.co_argcount]for n in arqumentNames:setattr(self,n,d[n])

通过获取__init__方法的代码对象,attributesFromArguments 函数能够只处理传递给__init__ 的参数名。因此你的__init__方法应当调用 attributesFromArguments(locals( )),而不是 attributesFromDict(locals()),在这个调用之后如果还有需要,可以继续加入更多代码并定义其他局部变量。

attributesFromArguments 最关键的限制是它不支持__init__ 的某种特殊参数,即kw。我们可以通过引入更多的内省来获得处理kw的能力,但那就要求使用更多的黑魔法和引入更多的复杂性,比起获得的这点功能似乎有些不值得。如果你确实想在这方面做些探索,为了实现内省,应当使用标准库的inspect模块,而不是在attributesFromArguments函数中自己实现这个功能。通过使用 inspect.getargspec(self.init),你能够同时知道参数名以及 self.__init__是否接受**kw 形式的参数。关于 insepectgetargspec 的更多信息请参看6.19 节。最后,请记住 Python 编程中的一个至理名言:“尽最用标准库搞定一切!”

http://www.xdnf.cn/news/3647.html

相关文章:

  • 多端定制系统开发:打造高效全平台覆盖的APP解决方案
  • Python爬虫(16)Python爬虫数据存储新维度:Redis Edge近端计算赋能实时数据处理革命
  • phpyun人才系统v7.0升级v7.1 开源vip版,php云专业人才招聘系统小程序零工市场源码支持v4.6的更新步骤流程详解
  • 工作记录 2015-07-15
  • 数据结构的基本概念以及算法的基本内容
  • python:如何获取股票 周K线数据、月K线数据
  • Go语言入门基础:协程
  • 【信息系统项目管理师-论文真题】2012上半年论文详解(包括解题思路和写作要点)
  • 装饰器@wraps(func)详解
  • 伊甸园之东: 农业革命与暴力的复杂性
  • Learning vtkjs之Cutter
  • 有向图强连通分量好题分享(一)
  • 【深度学习的灵魂】图片布局生成模型LayoutPrompt(2)·布局序列化模块
  • 如何通过文理工三类AI助理赋能HI,从而,颠覆“隔行如隔山”的旧观念和“十万小时定律”的成长限制
  • 快速掌握--cursor
  • 深入解析 Stacking:集成学习的“超级英雄联盟
  • 排查 EF 保存数据时提示:Validation failed for one or more entities 的问题
  • 以梦为舟,共赴中医星辰大海
  • 【Mytais系列】Type模块:源码
  • MySQL快速入门篇---数据库约束
  • 【计算机视觉】三维重建: MVSNet:基于深度学习的多视图立体视觉重建框架
  • 驱动精灵v9.7(含网卡版)驱动工具软件下载及安装教程
  • 360驱动大师v2.0(含网卡版)驱动工具软件下载及安装教程
  • The Traitor King (10 player 25 player)
  • 【网络编程】HTTP(超文本传输协议)详解
  • 文献总结:TPAMI端到端自动驾驶综述——End-to-End Autonomous Driving: Challenges and Frontiers
  • 《解锁Windows下GCC升级密码,开启高效编程新旅程》
  • 游戏引擎学习第255天:构建配置树
  • 【AI面试准备】Testim.io自动化测试工具使用指南
  • C++ 友元:打破封装的钥匙