python成长之路——第六天
定义
Python 的 Class 比较特别,和我们习惯的静态语言类型定义有很大区别。
1. 使用一个名为 __init__ 的方法来完成初始化。
2. 使用一个名为 __del__ 的方法来完成类似析购操作。
3. 所有的实例方法都拥有一个 self 参数来传递当前实例,类似于 this。
4. 可以使用 __class__ 来访问类型成员
def __init__(self):
print "initialize
def Foo(self):
print id(self)
>>>>>> a = MyClass()
initialize.
>>>>>> a.Foo()
14412576
>>>>>> id(a)
14412576
Class 有一些特殊的属性,便于我们获得一些额外的信息。
"""This is MyClass's Docoment"""
def __init__(self):
self.i = 1234
>>>>>> MyClass.__doc__ # 类型帮助信息
"This is MyClass's Docoment""This is MyClass's Docoment"
>>>>>> MyClass.__name__ # 类型名称
'MyClass'
>>>>>> MyClass.__module__ # 类型所在模块
'__main__'
>>>>>> MyClass.__bases__ # 类型所继承的基类(Python 支持多继承)
(<type 'object'>,)
>>>>>> MyClass.__dict__ # 类型字典,存储所有类型成员信息
<dictproxy object at 0x00DC1AD0>
>>>>>> #以下是实例拥有的属性
>>>>>> MyClass().__class__ # 实例的类型
<class '__main__.MyClass'>
>>>>>> MyClass().__module__ # 实例类型所在模块
'__main__'
>>>>>> MyClass().__dict__ # 对象字典,存储所有实例成员信息
{'i': 1234}
>>>>>>
继承
Python 支持多继承,但有几点需要注意:
1. 基类 __init__ / __del__ 需显示调用。
2. 继承方法的调用和基类声明顺序有关。
def __init__(self):
print "Base1"
def test(self):
print "Base1 test
>>>>>> class Base2:
def __init__(self):
print "Base2"
def test(self):
print "Base2 test"
>>>>>> class MyClass(Base2,Base1):
def __init__(self):
Base1.__init__(self)
Base2.__init__(self)
print "MyClass"
>>>>>> a = MyClass()
Base1
Base2
MyClass
>>>>>> a.test()
Base2 test
>>>>>> # 下面把 Base1 放在前面
>>>>>> class MyClass(Base1,Base2):
def __init__(self):
Base1.__init__(self)
Base2.__init__(self)
print "MyClass"
>>>>>> a = MyClass()
Base1
Base2
MyClass
>>>>>> a.test()
Base1 test
>>>>>>
成员
Python Class 同样包含类型和实例两种成员。
i = 123 # 类成员
def __init__(self):
self.i = 100 # 实例成员
>>>>>> print Class1.i
123
>>>>>> print Class1().i
100
>>>>>>
有几个很 "特殊" 的 "规则" 需要注意。
(1) 我们可以通过实例引用访问类型成员。因此下面的例子中 self.i 实际指向 Class1.i,直到我们为实例新增了一个成员 i。
>>>>>> class Class1:
i = 123
def __init__(self):
print self.i
print hex(id(self.i))
>>>>>> hex(id(Class1.i)) # 显示 Class1.i 的地址
'0xab5860'
>>>>>> a = Class1() # 创建 Class1 实例,我们会发现 self.i 实际指向 Class1.i
123
0xab5860
>>>>>> Class1.__dict__ # 显示 Class1 成员
{'i': 123, '__module__': '__main__', '__doc__': None, '__init__': <function __init__ at 0x012911B0>}
>>>>>> a.__dict__ # 显示实例成员
{}
>>>>>> a.i = 100 # 为实例新增加一个成员i
>>>>>> hex(id(a.i)) # 显示新成员i的地址
'0xab5974'
>>>>>> a.__dict__ # 显示实例成员
{'i': 100}
>>>>>>
(2) 调用类型内部方法,需要省略 self 参数。
def __init__(self):
self.__test("Hello Python")
def __test(self, s):
print s
>>>>>> Class1()
Hello Python
<__main__.Class1 instance at 0x00DC3800>
>>>>>>
我们可以在成员名称前添加 "__" 使其成为私有成员。
__i = 123
def __init__(self):
self.__x = 0
def __test(self):
print id(self)
>>>>>> Class1.i
Traceback (most recent call last):
File "<pyshell#203>", line 1, in <module>
Class1.i
AttributeError: class Class1 has no attribute 'i'
>>>>>> Class1().__x
Traceback (most recent call last):
File "<pyshell#204>", line 1, in <module>
Class1().__x
AttributeError: Class1 instance has no attribute '__x'
>>>>>> Class1().__test()
Traceback (most recent call last):
File "<pyshell#205>", line 1, in <module>
Class1().__test()
AttributeError: Class1 instance has no attribute '__test'
>>>>>>
事实上这只是一种规则,并不是编译器上的限制。我们依然可以用特殊的语法来访问私有成员。
123
>>>>>> a = Class1()
>>>>>> a._Class1__x
0
>>>>>> a._Class1__test()
14432256
>>>>>>
除了静态(类型)字段,我们还可以定义静态方法。
@staticmethod
def test():
print "In Static method
>>>>>> Class1.test()
In Static method
>>>>>>
从设计的角度,或许更希望用属性(property)来代替字段(field)。
def __init__(self):
self.__i = 1234
def getI(self): return self.__i
def setI(self, value): self.__i = value
def delI(self): del self.__i
I = property(getI, setI, delI, "Property I")
>>>>>> a = Class1()
>>>>>> a.I
1234
>>>>>> a.I = 1000
>>>>>> a.I
1000
如果只是 readonly property,还可以用另外一种方式。
def __init__(self):
self.__i = 1234
@property
def I(self):
return self.__i
>>>>>> a = Class1()
>>>>>> a.I
1234
用 __getitem__ 和 __setitem__ 可以实现 C# 索引器的功能。
def __init__(self):
self.__x = ["a", "b", "c"]
def __getitem__(self, key):
return self.__x[key]
def __setitem__(self, key, value):
self.__x[key] = value
>>>>>> a = Class1()
>>>>>> a[1]
'b'
>>>>>> a[1] = "xxx"
>>>>>> a[1]
'xxx'
>>>>>>
python成长之路——第六天的更多相关文章
- 我的Python成长之路---第六天---Python基础(20)---2016年2月20日(晴)
一.面向对象基础 面向对象名词解释: 类(Class): 用来描述具有相同的属性和方法的对象的集合.它定义了该集合中每个对象所共有的属性和方法.对象是类的实例. 类变量:类变量在整个实例化的对象中是公 ...
- 我的Python成长之路---第六天---Python基础(19)---2016年2月20日(晴)
shelve模块 shelve模块是pickle模块的扩展,可以通过key,value的方式访问pickle持久化保存的数据 持久化保存: 1 2 3 4 5 6 7 8 9 10 11 12 13 ...
- 我的Python成长之路---第六天---Python基础(18)---2016年2月20日(晴)
os模块 提供对操作系统进行调用的接口 >>> import os >>> os.getcwd() # 获取当前工作目录,类似linux的pwd命令 '/data/ ...
- 【Python成长之路】Python爬虫 --requests库爬取网站乱码(\xe4\xb8\xb0\xe5\xa)的解决方法【华为云分享】
[写在前面] 在用requests库对自己的CSDN个人博客(https://blog.csdn.net/yuzipeng)进行爬取时,发现乱码报错(\xe4\xb8\xb0\xe5\xaf\x8c\ ...
- (转)Python成长之路【第九篇】:Python基础之面向对象
一.三大编程范式 正本清源一:有人说,函数式编程就是用函数编程-->错误1 编程范式即编程的方法论,标识一种编程风格 大家学习了基本的Python语法后,大家就可以写Python代码了,然后每个 ...
- 【Python成长之路】装逼的一行代码:快速共享文件
[Python成长之路]装逼的一行代码:快速共享文件 2019-10-26 15:30:05 华为云 阅读数 335 文章标签: Python编程编程语言程序员Python开发 更多 分类专栏: 技术 ...
- python成长之路第三篇(1)_初识函数
目录: 函数 为什么要使用函数 什么是函数 函数的返回值 文档化函数 函数传参数 文件操作(二) 1.文件操作的步骤 2.文件的内置方法 函数: 一.为什么要使用函数 在日常写代码中,我们会发现有很多 ...
- 我的Python成长之路---第一天---Python基础(1)---2015年12月26日(雾霾)
2015年12月26日是个特别的日子,我的Python成之路迈出第一步.见到了心目中的Python大神(Alex),也认识到了新的志向相投的伙伴,非常开心. 尽管之前看过一些Python的视频.书,算 ...
- Python成长之路第二篇(1)_数据类型内置函数用法
数据类型内置函数用法int 关于内置方法是非常的多这里呢做了一下总结 (1)__abs__(...)返回x的绝对值 #返回x的绝对值!!!都是双下划线 x.__abs__() <==> a ...
随机推荐
- python初探-copy
python中,数据的拷贝有以下三种形式:赋值.浅copy和深copy.根据类型的不同,可以把数据分成以下两类:字符串和数字为一类,其他(包括列表.元祖.字典...)为一类. 在python中有池的概 ...
- 项目管理软件伙伴https://www.huobanyun.cn/
现在项目管理软件市面上很多,但能够完全适合每家公司需求的比较难找,因为众口难调,每家公司都有自己的特殊情况,所以,建议考虑下有比较齐全的基础功能的标准化软件产品,同时又在项目管理开发能力上比较突出. ...
- Sql:查看数据库表和表结构的语句
T-sql 显示表结构和字段信息的sql语句: exec sp_help tablename; ~~使用存储过程 sp_help 显示数据库包含哪些表的sql语句: use yourDBname;se ...
- php中 $$str 中 "$$" 的解释
原文:php中 $$str 中 "$$" 的解释 这种写法称为可变变量有时候使用可变变量名是很方便的.就是说,一个变量的变量名可以动态的设置和使用.一个普通的变量通过声明来设置,例 ...
- BZOJ 1089 严格n元树 (递推+高精度)
题解:用a[i]表<=i时有几种树满足度数要求,那么这样就可以递归了,a[i]=a[i-1]^n+1.n个节点每个有a[i-1]种情况,那么将其相乘,最后加上1,因为深度为0也算一种.那么答案就 ...
- HDU 2147 kiki's game
题解:画图可得当横纵坐标均为奇数时为必败态…… #include <cstdio> int main(){ int a,b; while(scanf("%d%d",&a ...
- 代码收藏 JS实现页内查找定位功能
前部分为IE下搜索方法 用TextRange来实现 后部分为firefox.chrome下搜索方法 var nextIndex = 0; var searchValue = ''; var input ...
- Flex 按钮添加图标
第一种是在Flex应用中创建一个变量,利用[Bindable]和[Embed] ,在代码中以参数形式传入制定图标(icon)的路径,然后利用类似icon="{Icon}"的代码嵌入 ...
- 人生新开始——第一天上班
今天是我正式上班的第一天,对今天要做的工作充满期待,对将要面对的同事们也充满期待.这天起的很早,7点钟就开始睡不着了,忙着起来整理东西,看着时间还早,便打开电脑听起CNN News,练习听力,一听就听 ...
- HDU 4725 The Shortest Path in Nya Graph-【SPFA最短路】
题目:http://acm.hdu.edu.cn/showproblem.php?pid=4725 题意:有N个点和N层..一层有X个点(0<=X<=N).两邻两层间有一条路花费C.还有M ...