面向对象编程
变量名类似__xxx__的,也就是以双下划线开头,并且以双下划线结尾的,是特殊变量,特殊变量是可以直接访问的,不是private变量,所以,不能用__name__、__score__这样的变量名。
有些时候,你会看到以一个下划线开头的实例变量名,比如_name,这样的实例变量外部是可以访问的,但是,按照约定俗成的规定,当你看到这样的变量时,意思就是,“虽然我可以被访问,但是,请把我视为私有变量,不要随意访问”。
但是强烈建议你不要这么干,因为不同版本的Python解释器可能会把__name改成不同的变量名。
总的来说就是,Python本身没有任何机制阻止你干坏事,一切全靠自觉。
表面上看,外部代码“成功”地设置了__name变量,但实际上这个__name变量和class内部的__name变量不是一个变量!内部的__name变量已经被Python解释器自动改成了_Student__name,而外部代码给bart新增了一个__name变量.
type,isinstance
isinstance([1, 2, 3], (list, tuple))
hasattr(obj, 'x') # 有属性'x'吗?
setattr(obj, 'y', 19) # 设置一个属性'y'
面向对象高级编程
s.name = 'Michael' # 动态给实例绑定一个属性
from types import MethodType
s.set_age = MethodType(set_age, s) # 给实例绑定一个方法
Student.set_score = set_score
class Student(object):
__slots__ = ('name', 'age') # 用tuple定义允许绑定的属性名称
__slots__定义的属性仅对当前类实例起作用,对继承的子类是不起作用的:
Python内置的@property装饰器就是负责把一个方法变成属性调用的
@property
def score(self):
return self._score
@score.setter
def score(self, value):
多重继承
__str__
__repr__()
__iter__
__next__(self):
__getitem__
__getitem__()传入的参数可能是一个int,也可能是一个切片对象slice
负数作处理
如果把对象看成dict,__getitem__()的参数也可能是一个可以作key的object,例如str。
与之对应的是__setitem__()方法,把对象视作list或dict来对集合赋值。最后,还有一个__delitem__()方法,用于删除某个元素。
__getattr__
__call__
>>> s = Student('Michael')
>>> s() # self参数不要传入
My name is Michael.
type()
>>> Hello = type('Hello', (object,), dict(hello=fn)) # 创建Hello class
>>> h = Hello()
>>> h.hello()
metaclass是Python面向对象里最难理解,也是最难使用的魔术代码。正常情况下,你不会碰到需要使用metaclass的情况,所以,以下内容看不懂也没关系,因为基本上你不会用到。
错误,调试,单元测试,文档测试