1. 文章转载自廖雪峰老师Python课程博客,仅供学习参考使用

    看到类似__slots__这种形如__xxx__的变量或者函数名就要注意,这些在Python中是有特殊用途的。
  2.  
  3. __slots__我们已经知道怎么用了,__len__()方法我们也知道是为了能让class作用于len()函数。
  4.  
  5. 除此之外,Pythonclass中还有许多这样有特殊用途的函数,可以帮助我们定制类。
  6.  
  7. __str__
  8. 我们先定义一个Student类,打印一个实例:
  9.  
  10. >>> class Student(object):
  11. ... def __init__(self, name):
  12. ... self.name = name
  13. ...
  14. >>> print(Student('Michael'))
  15. <__main__.Student object at 0x109afb190>
  16. 打印出一堆<__main__.Student object at 0x109afb190>,不好看。
  17.  
  18. 怎么才能打印得好看呢?只需要定义好__str__()方法,返回一个好看的字符串就可以了:
  19.  
  20. >>> class Student(object):
  21. ... def __init__(self, name):
  22. ... self.name = name
  23. ... def __str__(self):
  24. ... return 'Student object (name: %s)' % self.name
  25. ...
  26. >>> print(Student('Michael'))
  27. Student object (name: Michael)
  28. 这样打印出来的实例,不但好看,而且容易看出实例内部重要的数据。
  29.  
  30. 但是细心的朋友会发现直接敲变量不用print,打印出来的实例还是不好看:
  31.  
  32. >>> s = Student('Michael')
  33. >>> s
  34. <__main__.Student object at 0x109afb310>
  35. 这是因为直接显示变量调用的不是__str__(),而是__repr__(),两者的区别是__str__()返回用户看到的字符串,而__repr__()返回程序开发者看到的字符串,也就是说,__repr__()是为调试服务的。
  36.  
  37. 解决办法是再定义一个__repr__()。但是通常__str__()和__repr__()代码都是一样的,所以,有个偷懒的写法:
  38.  
  39. class Student(object):
  40. def __init__(self, name):
  41. self.name = name
  42. def __str__(self):
  43. return 'Student object (name=%s)' % self.name
  44. __repr__ = __str__
  45. __iter__
  46. 如果一个类想被用于for ... in循环,类似listtuple那样,就必须实现一个__iter__()方法,该方法返回一个迭代对象,然后,Pythonfor循环就会不断调用该迭代对象的__next__()方法拿到循环的下一个值,直到遇到StopIteration错误时退出循环。
  47.  
  48. 我们以斐波那契数列为例,写一个Fib类,可以作用于for循环:
  49.  
  50. class Fib(object):
  51. def __init__(self):
  52. self.a, self.b = 0, 1 # 初始化两个计数器a,b
  53.  
  54. def __iter__(self):
  55. return self # 实例本身就是迭代对象,故返回自己
  56.  
  57. def __next__(self):
  58. self.a, self.b = self.b, self.a + self.b # 计算下一个值
  59. if self.a > 100000: # 退出循环的条件
  60. raise StopIteration()
  61. return self.a # 返回下一个值
  62. 现在,试试把Fib实例作用于for循环:
  63.  
  64. >>> for n in Fib():
  65. ... print(n)
  66. ...
  67. 1
  68. 1
  69. 2
  70. 3
  71. 5
  72. ...
  73. 46368
  74. 75025
  75. __getitem__
  76. Fib实例虽然能作用于for循环,看起来和list有点像,但是,把它当成list来使用还是不行,比如,取第5个元素:
  77.  
  78. >>> Fib()[5]
  79. Traceback (most recent call last):
  80. File "<stdin>", line 1, in <module>
  81. TypeError: 'Fib' object does not support indexing
  82. 要表现得像list那样按照下标取出元素,需要实现__getitem__()方法:
  83.  
  84. class Fib(object):
  85. def __getitem__(self, n):
  86. a, b = 1, 1
  87. for x in range(n):
  88. a, b = b, a + b
  89. return a
  90. 现在,就可以按下标访问数列的任意一项了:
  91.  
  92. >>> f = Fib()
  93. >>> f[0]
  94. 1
  95. >>> f[1]
  96. 1
  97. >>> f[2]
  98. 2
  99. >>> f[3]
  100. 3
  101. >>> f[10]
  102. 89
  103. >>> f[100]
  104. 573147844013817084101
  105. 但是list有个神奇的切片方法:
  106.  
  107. >>> list(range(100))[5:10]
  108. [5, 6, 7, 8, 9]
  109. 对于Fib却报错。原因是__getitem__()传入的参数可能是一个int,也可能是一个切片对象slice,所以要做判断:
  110.  
  111. class Fib(object):
  112. def __getitem__(self, n):
  113. if isinstance(n, int): # n是索引
  114. a, b = 1, 1
  115. for x in range(n):
  116. a, b = b, a + b
  117. return a
  118. if isinstance(n, slice): # n是切片
  119. start = n.start
  120. stop = n.stop
  121. if start is None:
  122. start = 0
  123. a, b = 1, 1
  124. L = []
  125. for x in range(stop):
  126. if x >= start:
  127. L.append(a)
  128. a, b = b, a + b
  129. return L
  130. 现在试试Fib的切片:
  131.  
  132. >>> f = Fib()
  133. >>> f[0:5]
  134. [1, 1, 2, 3, 5]
  135. >>> f[:10]
  136. [1, 1, 2, 3, 5, 8, 13, 21, 34, 55]
  137. 但是没有对step参数作处理:
  138.  
  139. >>> f[:10:2]
  140. [1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89]
  141. 也没有对负数作处理,所以,要正确实现一个__getitem__()还是有很多工作要做的。
  142.  
  143. 此外,如果把对象看成dict__getitem__()的参数也可能是一个可以作keyobject,例如str
  144.  
  145. 与之对应的是__setitem__()方法,把对象视作listdict来对集合赋值。最后,还有一个__delitem__()方法,用于删除某个元素。
  146.  
  147. 总之,通过上面的方法,我们自己定义的类表现得和Python自带的listtupledict没什么区别,这完全归功于动态语言的“鸭子类型”,不需要强制继承某个接口。
  148.  
  149. __getattr__
  150. 正常情况下,当我们调用类的方法或属性时,如果不存在,就会报错。比如定义Student类:
  151.  
  152. class Student(object):
  153.  
  154. def __init__(self):
  155. self.name = 'Michael'
  156. 调用name属性,没问题,但是,调用不存在的score属性,就有问题了:
  157.  
  158. >>> s = Student()
  159. >>> print(s.name)
  160. Michael
  161. >>> print(s.score)
  162. Traceback (most recent call last):
  163. ...
  164. AttributeError: 'Student' object has no attribute 'score'
  165. 错误信息很清楚地告诉我们,没有找到score这个attribute
  166.  
  167. 要避免这个错误,除了可以加上一个score属性外,Python还有另一个机制,那就是写一个__getattr__()方法,动态返回一个属性。修改如下:
  168.  
  169. class Student(object):
  170.  
  171. def __init__(self):
  172. self.name = 'Michael'
  173.  
  174. def __getattr__(self, attr):
  175. if attr=='score':
  176. return 99
  177. 当调用不存在的属性时,比如scorePython解释器会试图调用__getattr__(self, 'score')来尝试获得属性,这样,我们就有机会返回score的值:
  178.  
  179. >>> s = Student()
  180. >>> s.name
  181. 'Michael'
  182. >>> s.score
  183. 99
  184. 返回函数也是完全可以的:
  185.  
  186. class Student(object):
  187.  
  188. def __getattr__(self, attr):
  189. if attr=='age':
  190. return lambda: 25
  191. 只是调用方式要变为:
  192.  
  193. >>> s.age()
  194. 25
  195. 注意,只有在没有找到属性的情况下,才调用__getattr__,已有的属性,比如name,不会在__getattr__中查找。
  196.  
  197. 此外,注意到任意调用如s.abc都会返回None,这是因为我们定义的__getattr__默认返回就是None。要让class只响应特定的几个属性,我们就要按照约定,抛出AttributeError的错误:
  198.  
  199. class Student(object):
  200.  
  201. def __getattr__(self, attr):
  202. if attr=='age':
  203. return lambda: 25
  204. raise AttributeError('\'Student\' object has no attribute \'%s\'' % attr)
  205. 这实际上可以把一个类的所有属性和方法调用全部动态化处理了,不需要任何特殊手段。
  206.  
  207. 这种完全动态调用的特性有什么实际作用呢?作用就是,可以针对完全动态的情况作调用。
  208.  
  209. 举个例子:
  210.  
  211. 现在很多网站都搞REST API,比如新浪微博、豆瓣啥的,调用APIURL类似:
  212.  
  213. http://api.server/user/friends
  214. http://api.server/user/timeline/list
  215. 如果要写SDK,给每个URL对应的API都写一个方法,那得累死,而且,API一旦改动,SDK也要改。
  216.  
  217. 利用完全动态的__getattr__,我们可以写出一个链式调用:
  218.  
  219. class Chain(object):
  220.  
  221. def __init__(self, path=''):
  222. self._path = path
  223.  
  224. def __getattr__(self, path):
  225. return Chain('%s/%s' % (self._path, path))
  226.  
  227. def __str__(self):
  228. return self._path
  229.  
  230. __repr__ = __str__
  231. 试试:
  232.  
  233. >>> Chain().status.user.timeline.list
  234. '/status/user/timeline/list'
  235. 这样,无论API怎么变,SDK都可以根据URL实现完全动态的调用,而且,不随API的增加而改变!
  236.  
  237. 还有些REST API会把参数放到URL中,比如GitHubAPI
  238.  
  239. GET /users/:user/repos
  240. 调用时,需要把:user替换为实际用户名。如果我们能写出这样的链式调用:
  241.  
  242. Chain().users('michael').repos
  243. 就可以非常方便地调用API了。有兴趣的童鞋可以试试写出来。
  244.  
  245. __call__
  246. 一个对象实例可以有自己的属性和方法,当我们调用实例方法时,我们用instance.method()来调用。能不能直接在实例本身上调用呢?在Python中,答案是肯定的。
  247.  
  248. 任何类,只需要定义一个__call__()方法,就可以直接对实例进行调用。请看示例:
  249.  
  250. class Student(object):
  251. def __init__(self, name):
  252. self.name = name
  253.  
  254. def __call__(self):
  255. print('My name is %s.' % self.name)
  256. 调用方式如下:
  257.  
  258. >>> s = Student('Michael')
  259. >>> s() # self参数不要传入
  260. My name is Michael.
  261. __call__()还可以定义参数。对实例进行直接调用就好比对一个函数进行调用一样,所以你完全可以把对象看成函数,把函数看成对象,因为这两者之间本来就没啥根本的区别。
  262.  
  263. 如果你把对象看成函数,那么函数本身其实也可以在运行期动态创建出来,因为类的实例都是运行期创建出来的,这么一来,我们就模糊了对象和函数的界限。
  264.  
  265. 那么,怎么判断一个变量是对象还是函数呢?其实,更多的时候,我们需要判断一个对象是否能被调用,能被调用的对象就是一个Callable对象,比如函数和我们上面定义的带有__call__()的类实例:
  266.  
  267. >>> callable(Student())
  268. True
  269. >>> callable(max)
  270. True
  271. >>> callable([1, 2, 3])
  272. False
  273. >>> callable(None)
  274. False
  275. >>> callable('str')
  276. False
  277. 通过callable()函数,我们就可以判断一个对象是否是“可调用”对象。

Python基础(定制类)的更多相关文章

  1. python基础——定制类

    python基础——定制类 看到类似__slots__这种形如__xxx__的变量或者函数名就要注意,这些在Python中是有特殊用途的. __slots__我们已经知道怎么用了,__len__()方 ...

  2. python基础——枚举类

    python基础——枚举类 当我们需要定义常量时,一个办法是用大写变量通过整数来定义,例如月份: JAN = 1 FEB = 2 MAR = 3 ... NOV = 11 DEC = 12 好处是简单 ...

  3. Python基础-类的探讨(class)

    Python基础-类的探讨(class) 我们下面的探讨基于Python3,我实际测试使用的是Python3.2,Python3与Python2在类函数的类型上做了改变 1,类定义语法  Python ...

  4. 二十六. Python基础(26)--类的内置特殊属性和方法

    二十六. Python基础(26)--类的内置特殊属性和方法 ● 知识框架 ● 类的内置方法/魔法方法案例1: 单例设计模式 # 类的魔法方法 # 案例1: 单例设计模式 class Teacher: ...

  5. 快速了解Python的定制类

    多重继承 class Student(man,oldman): pass 可以继承多个父类,拥有他们的方法,如果有父类有相同的方法,哪个在前用哪个 定制类 看到类似__slots__这种形如 __xx ...

  6. python基础(26):类的成员(字段、方法、属性)

    1. 字段 字段:包括普通字段和静态字段,他们在定义和使用中有所区别,而最本质的区别是内存中保存的位置不同. 普通字段属于对象 静态字段属于类 字段的定义和使用: class Province: # ...

  7. Python基础(十一) 类继承

    类继承: 继承的想法在于,充份利用已有类的功能,在其基础上来扩展来定义新的类. Parent Class(父类) 与 Child Class(子类): 被继承的类称为父类,继承的类称为子类,一个父类, ...

  8. python基础----元类metaclass

    1 引子 class Foo: pass f1=Foo() #f1是通过Foo类实例化的对象 python中一切皆是对象,类本身也是一个对象,当使用关键字class的时候,python解释器在加载cl ...

  9. python基础编程——类和实例

    在了解类和实例之前,需要先了解什么是面向对象,什么又是面向过程.面向过程是以过程为中心实现一步步操作(相互调用,类似流水线思想):面向对象是以事物为中心,某个事物可以拥有自己的多个行为,而另一个事物也 ...

  10. Python基础(9) - 类

    Python 看下面一个简单类: >>> class MyClass(object): ... """ ... this is a class with ...

随机推荐

  1. FastAPI(44)- 操作关系型数据库

    ORM FastAPI 可与任何数据库和任何样式的库配合使用并和数据库通信 object-relational mapping 对象关系映射 ORM 具有在代码和数据库表(关系)中的对象之间进行转换( ...

  2. Serverless 的初心、现状和未来

    作者 | 不瞋 导读:Serverless 是如何产生的?当前有哪些落地场景?Serverless 的未来又将如何?本文分享了阿里云高级技术专家不瞋对于 Serverless 的看法,回顾其发展历程, ...

  3. 详解package-lock.json的作用

    目录 详解package-lock.json package-lock.json的作用 版本号的定义规则与前缀对安装的影响 改动package.json后依旧能改变项目依赖的版本 当前项目的真实版本号 ...

  4. Tomcat各种日志的关系与catalina.out文件的分割

    Tomcat 各日志之间的关系 一图胜千言! 其他日志如localhost.{yyyy-MM-dd}.log.localhost-access.{yyyy-MM-dd}.log是context的名称, ...

  5. SpringBoot配置文件application

    配置文件 SpringBoot使用一个全局的配置文件 , 配置文件名称是固定的,有两种文件格式: application.properties 语法结构 :key=value application. ...

  6. Java(5)输入和输出

    作者:季沐测试笔记 原文地址:https://www.cnblogs.com/testero/p/15201515.html 博客主页:https://www.cnblogs.com/testero ...

  7. vue基础-组件&插槽

    组件 组件化的意义:封装(复用,把逻辑隐藏起来,提高可维护性),快速开发(搭积木) 约定:我们通常把那些除了HTML标签以外的自定义组件,才称为'组件',结论是,我们说"父组件"& ...

  8. PinPoint单节点部署及客户端配置方法

    在一次做项目中,需要涉及全链路压测,为了更好定位链路中某一节点可能会出现的问题,在繁忙之余,快速部署及应用了该链路工具,分享给大家~ 话不多说,开始部署~ 一.环境配置1.1 获取需要的依赖包进入ho ...

  9. Java:容器类线程不安全

    Java:容器类线程不安全 本笔记是根据bilibili上 尚硅谷 的课程 Java大厂面试题第二季 而做的笔记 1. Collection 线程不安全的举例 前言 1.当我们执行下面语句的时候,底层 ...

  10. Alpha-功能规格说明书

    项目 内容 这个作业属于哪个课程 2021春季软件工程(罗杰 任健) 这个作业的要求在哪里 团队项目-计划-功能规格说明书 一.引言 1. 项目简介 项目团队:删库跑路对不队 项目名称:题士 项目内容 ...