解释

我们通常把 __init__ 称为构造方法,这是从其他语言借鉴过来的术语。

其实,用于构建实例的是特殊方法 __new__:这是个类方法(使用特殊方式处理,因此不必使用 @classmethod 装饰器),必须返回一个实例。返回的实例会作为第一个参数(即 self)传给 __init__ 方法。

因为调用 __init__ 方法时要传入实例,而且禁止返回任何值,所以 __init__ 方法其实是“初始化方法”。真正的构造方法是 __new__

我们几乎不需要自己编写 __new__ 方法,因为从 object 类继承的实现已经足够了。刚才说明的过程,即从__new__ 方法到__init__ 方法,是最常见的,但不是唯一的。 __new__ 方法也可以返回其他类的实例,此时,解释器不会调用 __init__ 方法。

也就是说,Python 构建对象的过程可以使用下述伪代码概括:

# 构建对象的伪代码
def object_maker(the_class, some_arg):
new_object = the_class.__new__(some_arg)
if isinstance(new_object, the_class):
the_class.__init__(new_object, some_arg)
return new_object # 下述两个语句的作用基本等效
x = Foo('bar')
x = object_maker(Foo, 'bar')

示例:对JSON的解析

import keyword

class FrozenJSON(object):
"""一个只读接口
使用属性表示访问JSON类对象
""" def __init__(self, mapping):
self._data = {}
for k, v in mapping.items():
if keyword.iskeyword(k):
k += '_'
self._data[k] = v def __getattr__(self, name):
if hasattr(self._data, name):
return getattr(self._data, name)
else:
return FrozenJSON.build(self._data[name]) @classmethod
def build(cls, obj):
if isinstance(obj, abc.Mapping):
return cls(obj)
elif isinstance(obj, abc.MutableSequence):
return [cls.build(item) for item in obj]
else:
return obj class FrozenJSON2(object):
"""通过定义 __new__ 方法完成实例创建时的行为"""
def __new__(cls, arg):
if isinstance(arg, abc.Mapping):
return super().__new__(cls)
elif isinstance(arg, abc.MutableSequence):
return [cls(item) for item in arg]
else:
return arg def __init__(self, mapping):
self._data = {}
for k, v in mapping.items():
if keyword.iskeyword(k):
k += '_'
self._data[k] = v def __getattr__(self, name):
if hasattr(self._data, name):
return getattr(self._data, name)
else:
return FrozenJSON(self._data[name])

摘自《流畅的Python》 19.1.3 使用__new__方法以灵活的方式创建对象

最新文章

  1. IOCP Internals
  2. vim编辑器的使用
  3. java线性表学习笔记(一)
  4. java_redis3.0.3集群搭建
  5. CSS链接、光标、DHTML、缩放
  6. windows上SVN服务器以及客户端TortoiseSVN的安装配置
  7. 项目中的报错信息,maven报错等的总结
  8. spring-oauth-server实践:使用授权方式四:client_credentials 模式下access_token做业务!!!
  9. php 二维数组根据值进行排序
  10. tshark的抓包和解析
  11. too much recursion(太多递归)Uncaught RangeError: Maximum call stack size exceeded BootstrapValidator报错
  12. springboot-aop
  13. centos磁盘挂载|centos虚拟机硬盘不够怎么办?|centos虚拟机硬盘的扩展
  14. JavaScriptDOM操作那些事儿
  15. 【LCA+MST】BZOJ3732-Network
  16. Robot Framework 自定义库
  17. Exp2 后门原理与实践 20164321 王君陶
  18. AngularJS学习笔记(2)——与用户交互的动态清单列表
  19. PHP视频教程 字符串处理函数(二)
  20. JPA criteria 查询:类型安全与面向对象

热门文章

  1. [hdu6316]Odd shops
  2. 智能 Request 推荐,K8s 资源利用率提升 252%
  3. mybatis判断集合长度
  4. 学习 NPM 最基础的指令
  5. 『学了就忘』Linux文件系统管理 — 58、常用硬盘管理相关命令
  6. Codeforces 979E Kuro and Topological Parity(dp)
  7. linux 线程函数小结
  8. Go语言核心36讲(Go语言实战与应用二十一)--学习笔记
  9. 截取字符串、拼接字符串【c#】
  10. java类加载、对象创建过程