《Python基础教程(第二版)》学习笔记 -> 第九章 魔法方法、属性和迭代器
准备工作
>>> class NewStyle(object): more_code_here >>> class OldStyle: more_code_here
在这两个类中,NewStyle是新式的类,OldStyle是旧式的类,如果文件以__metaclass__ = type 开始,那么两个类都是新式类。
构造方法
构造方法,当一个对象被创建后,会立即调用构造方法。Python中创建一个构造方法,只要把init方法的名字从简单的init修改成__init__ :
>>> class Foobar: def __init__(self): self.somevar = 42 >>> f = Foobar() >>> f.somevar 42
- 重写一般方法和特殊的构造方法
每个类都可能拥有一个或者多个超类,它们会从超类那里继承行为方式。如果一个方法在B类的一个实例中被调用,但在B类中没有找到该方法,那么就会去它的超类A里面找。考虑下面的两个类:class A: def hello(self): print "Hello, A!" class B(A): pass
A类定义了一个叫做hello的方法,被B类继承:
>>> a=A() >>> b = B() >>> a.hello() Hello, A! >>> b.hello() Hello, A!
使用这个定义,b.hello()能产生一个不同的结果
>>> class B(A): def hello(self): print "B" >>> b = B() >>> b.hello() B
重写是继承机制中一个重要内容。
下例:
class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaaah...' self.hungry = False else: print 'No thx!' pass class SongBird(Bird): def __init__(self): self.sound = 'Squawk!' def sing(self): print self.sound
运行:
>>> sb = SongBird() >>> sb.sing() Squawk! >>> sb.eat() Traceback (most recent call last): File "<pyshell#2>", line 1, in <module> sb.eat() File "C:/Users/User/Desktop/Python_Demo/c_3.py", line 5, in eat if self.hungry: AttributeError: SongBird instance has no attribute 'hungry'
异常很清楚地说明了错误:SongBird没有hungry特性。原因是在SongBird中,构造方法被重写,没有初始化hungry特性的代码。
- 调用未绑定的超类构造方法
写解决上例的问题,代码如下:class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaaah...' self.hungry = False else: print 'No thx!' pass class SongBird(Bird): def __init__(self): Bird.__init__(self) #banding method self.sound = 'Squawk!' def sing(self): print self.sound
运行结果:
>>> sb = SongBird() >>> sb.eat() Aaaaah... >>> sb.eat() No thx!
为什么会有注意的结果?在调用一个实例的方法时,该方法的self参数会被自动绑定到实例上(这被称为绑定方法)。但如果直接调用类的方法(比如Bird.__init__),那么就没有实例会被绑定。这样就可以自由地提供需要的self参数。这样的方法称为未绑定(unbound)方法。
- 使用super函数
__metaclass__ = type class Bird: def __init__(self): self.hungry = True def eat(self): if self.hungry: print 'Aaaaah...' self.hungry = False else: print 'No thx!' pass class SongBird(Bird): def __init__(self): super(SongBird,self).__init__() self.sound = 'Squawk!' def sing(self): print self.sound
运行结果如下:
>>> sb = SongBird() >>> sb.eat() Aaaaah... >>> sb.eat() No thx!
成员访问
- 基本的序列和映射规则
序列和映射是对象的集合。为了实现它们基本的行为,如果对象是不可变的,那么就需要使用两个魔法方法,如果是可变的则需要使用4个。
__len__(self):这个方法应该返回集合中所含项目的数量。(序列:返回个数;映射:返回键-值对的数量;返回0,对象会被当做一个布尔变量中的假值进行处理)
__getitem__(self.key):这个方法返回与所给键对应的值。对于一个序列,键应该是一个0~n-1的整数(也可能是负数,n是序列的长度);对于映射来说,可以使用任何种类的键。
__setitem__(self,key,value):这个方法应该按一定的方式存储和key相关的value,该值随后可使用__getitem__来获取。
__delitem__(self,key):这个方法对一部分对象使用del语句时被调用,同事必须删除和元素相关的键。 - 子类化列表、字典和字符串
属性
- property 函数
__metaclass__ = type class Rectangle: def __init__(self): self.width = 0 self.height = 0 def setSize(self,size): self.width,self.height = size def getSize(self): return self.width,self.height size = property(getSize,setSize)
运行结果:
>>> r = Rectangle() >>> r.width = 10 >>> r.height = 5 >>> r.size (10, 5) >>> r.size = 150,100 >>> r.height 100
size特性仍然取决于getSize和setSize中的计算。
迭代器
- 迭代器规则
迭代的意思是重复做一些事很多次。
__iter__方法返回一个迭代器(iterator),所谓的迭代器就是具有next方法的对象。在调用next方法时,迭代器会返回它的下一个值。如果next方法被调用,但迭代器没有值可以返回,就会引发一个StopIteration异常。class Fibs: def __init__(self): self.a = 0 self.b = 1 def next(self): self.a,self.b = self.b,self.a + self.b return self.a def __iter__(self): return self fibs = Fibs() for f in fibs: if f>1000: print f break
- 从迭代器得到序列
除了在迭代器和可迭代对象上进行迭代外,还能把它们转换为序列:class TestIterator: value = 0 def next(self): self.value += 1 if self.value >10:raise StopIteration return self.value def __iter__(self): return self
运行结果:
>>> ti = TestIterator() >>> list(ti) [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
生成器
生成器也叫简单生成器。
生成器是一种用普通的函数语法定义的迭代器。
- 创建生成器
>>> nested = [[1,2],[3,4],[5]] >>> def flatten(nested): for sublist in nested: for element in sublist: yield element >>> for num in flatten(nested): print num 1 2 3 4 5
任何包含yield语句的函数称为生成器。
- 递归生成器
- 通用生成器
生成器是由两部分组成:生成器的函数和生成器的迭代器。
生成器的函数是用def语句定义的,包含yield的部分,生成器的迭代器是这个函数的返回部分 - 生成器方法
- 模拟生成器
(目前难以结合实例来学习,故此文档笔记暂时高于段落,每天开始研究自动化测试,Python核心编程课后习题)
《Python基础教程(第二版)》学习笔记 -> 第九章 魔法方法、属性和迭代器的更多相关文章
- Jquery基础教程第二版学习记录
本文仅为个人jquery基础的学习,简单的记录以备忘. 在线手册:http://www.php100.com/manual/jquery/第一章:jquery入门基础jquery知识:jquery能做 ...
- 《Python基础教程》 读书笔记 第九章 魔法方法、属性和迭代器(上)
构造方法 在Python中创建一个构造方法很容易.只要把init方法的名字从简单的init修改为魔法版本__init__即可: >>> class FooBar: ... d ...
- <<Python基础教程>>学习笔记 | 第10章 | 充电时刻
第10章 | 充电时刻 本章主要介绍模块及其工作机制 ------ 模块 >>> import math >>> math.sin(0) 0.0 模块是程序 一个简 ...
- <<Python基础教程>>学习笔记 | 第04章 | 字典
第04章:字典 当索引不好用时 Python唯一的内建的映射类型,无序,但都存储在一个特定的键中.键能够使字符.数字.或者是元祖. ------ 字典使用: 表征游戏棋盘的状态,每一个键都是由坐标值组 ...
- <<Python基础教程>>学习笔记 | 第12章 | 图形用户界面
Python支持的工具包非常多.但没有一个被觉得标准的工具包.用户选择的自由度大些.本章主要介绍最成熟的跨平台工具包wxPython.官方文档: http://wxpython.org/ ------ ...
- 第二章、元组和列表(python基础教程第二版 )
最基本的数据结构是序列,序列中每个元素被分配一个序号-元素的位置,也称索引.第一个索引为0,最后一个元素索引为-1. python中包含6种内建的序列:元组.列表.字符串.unicode字符串.buf ...
- python基础教程第二版 第一章
1.模块导入python以增强其功能的扩展:三种方式实现 (1). >>> Import math >>> math.floor(32.9) 32.0 #按照 模块 ...
- <<Python基础教程>>学习笔记 | 第11章 | 文件和素材
打开文件 open(name[mode[,buffing]) name: 是强制选项,模式和缓冲是可选的 #假设文件不在.会报以下错误: >>> f = open(r'D:\text ...
- Docker技术入门与实战 第二版-学习笔记-10-Docker Machine 项目-2-driver
1>使用的driver 1〉generic 使用带有SSH的现有VM/主机创建机器. 如果你使用的是机器不直接支持的provider,或者希望导入现有主机以允许Docker Machine进行管 ...
随机推荐
- stringgird中使用TClientDataSet排序的问题
function TfrmMain.createIIReport(cdsBody: TClientDataSet; silent: Boolean): String;var s,sText: ...
- lintcode:Binary Tree Postorder Traversal 二叉树的后序遍历
题目: 二叉树的后序遍历 给出一棵二叉树,返回其节点值的后序遍历. 样例 给出一棵二叉树 {1,#,2,3}, 1 \ 2 / 3 返回 [3,2,1] 挑战 你能使用非递归实现么? 解题: 递归程序 ...
- java代码实现自动登录功能
通常我们登录某网站,会有选择保存几天,或者是几个星期不用登录,之后输入该网站地址无需登录直接进入主页面,那么这就叫做自动登录,怎么实现呢,下面我以一个小例子来演示一下 登录页面:login.jsp & ...
- Sina App Engine(SAE)入门教程(3)-KVDB使用
简介 因为传统关系型数据库在分布式环境下表现的扩展性不足等缺点,近年来NoSQL的概念渐渐成为业界关注的焦点,越来越多的技术人员也习惯于使用NoSQL数据库进行日常开发,SAE为了应对这种新需求,也进 ...
- 258. Add Digits
题目: Given a non-negative integer num, repeatedly add all its digits until the result has only one di ...
- ios7新增基础类库以及OC新特性
新特性: Modules:用XCode5新建工程默认支持modules编译,老项目需在Build Settings里查找modules,找到的Enable Modules选项设置为YES. 对应新增语 ...
- Ubuntu下的svn的安装
安装SVN问题很多,现在目前遇到的问题是,安装时候找不到svn connector的连接器 导致不能够对SVN插件进行完整安装.但是可以单独安装该插件 http://community.pol ...
- [HDOJ4027]Can you answer these queries?(线段树,特殊成段更新,成段查询)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4027 RT,该题要求每次更新是更新所有节点,分别求平方根,查询是求和.昨晚思前想后找有没有一个数学上的 ...
- abstract的method是否可同时是static,是否可同时是native,是否可同时是synchronized?
abstract的method不可以是static的,因为抽象的方法是要被子类实现的,而static与子类扯不上关系! native方法表示该方法要用另外一种依赖平台的编程语言实现的,不存在着被子类实 ...
- uva12716GCD XOR
筛法,打表. 通过打表可知,但gcd(a,b)==a xor b时,a xor b = a-b. 就是求满足 c = a-b且c = a xor b 的c的个数. #include<cstdio ...