new()
当使用"类名([实参])"创建实例对象时。Python解释器的主要处理过程包括两大步:
1.调用特殊方法__new__()创建实例对象
首先会查找该类对象是否实现了特殊方法__new__().
如果没有实现。则去其父类中依次查找,直到类对象object。
2.调用特殊方法__init__()对创建的实例对象进行初始化
new()返回的实例对象会作为实参被自动传递给__init__()的第一个形参self。
call()与callable()
如果在类对象中实现了特殊方法__call__ (), 那么就可以像调用函数一样直接调用这个类对象的实例对象,从而会自动调用特殊方法call_ ()。
class MyClass(object) :
def call (self, *args, **kwargs) :
print (args, kwargs)
mc= MyClass()
mc() #返回结果:() {}
mc(1,2,x=3,y=4) #返回结果:(1, 2) {‘x’: 3,‘y’: 4]
doc_与_dict
-
调用内置函数dir得到的类对象的所有属性中,有一个特殊属性叫__doc__, 用于表示类对象的文档字符串。
-
类对象的文档字符串
与函数的文档字符串类似,位于类对象的第一行的字符串被称为类对象的文档字符串,通常用三个引号表示。 类对象的文档字符串是对类对象的功能的简要描述。 -
访问类对象的文档字符串
通过类对象的特殊属性__doc__ 可以访问类对象的文档字符串。 调用内置函数help()得到的帮助信息中会包含类对象的文档字符串。class MyClass(object):
“”“这是类对象的文档字符串
1
2
3
“””
pass
print (MyClass.doc)
返回结果:
这是类对象的文档字符串
1
2
3
特殊属性之__slots__
Python是动态语言,所以,在创建对象之后,可以对其动态地绑定属性和方法。
如果想要对实例对象动态绑定的属性和方法的名称进行限制,可以在其对应的类对象中定义特殊属性__ slots__, 并给__slots__ 赋值一个所有元素都为字符串的列表或元组,这样,对实例对象动态绑定的属性和方法的名称就只能来自于__slots__ 中的元素。
class MyClass (object):
__slots__ = ("attrl", "do_sthl")
mc = MyClass() #返回结果:mc. attr1 = 18
print (mc. attr1) #返回结果:18
mc. attr2 = 56 #返回结果:错误
def do_sth1(self):
print("do_sth1被调用了")
from types import MethodType
mc. do_sth1 = MethodType(do_sth1, mc)
mc. do_ sth1() #返回结果:do_sth1被调用了
def do_sth2(self):
print("do_sth2被调用了")
mc. do_sth2 = MethodType(do_sth2, mc) #返回结果:错误
**默认情况下,访问实例对象的属性是通过访问该实例对象的特殊属性__dict__ 来实现的。例如:访问obj.x其实访问的是obj.dict [‘x’]
在类对象中定义了特殊属性__slots__后,其实例对象就不会在创建特殊属性__dict__ 了,而是为每个属性创建一个描述器,访问属性时就会直接调用这个描述器。调用描述器比访问__dict__ 要快,因此,在类对象中定义特殊属性__slots__ 可 以提高属性的访问速度。
此外,在类对象中定义了特殊属性__slots__ 后, 由于其实例对象不再创建特殊属性__dict__ , 同时,特殊属性__dict__ 是一个字典,字典的本质是哈希表,是一种用空间换取时间的数据结构,因此,在类对象中定义特殊属性__ slots__ 可以减少内存消耗。**
- 特殊属性__slots__只对其所在类对象的实例对象起作用,对其所在类对象的子类的実例实例对象是不起作用的。
- 如果子类也定义了特殊属性__slots__,那么子类的实例对象可以动态绑定的属性和方法的名称为子类的__slots__ 加上父类的__
slots__。
特殊方法之__len__()
内置函数len( )用于返回对象的长度。
内置函数len()的实参在默认情况下不能是自定义类对象的实例对象。
print(len([1,2,3, 4, 5])) #5
print(len('abcde' )) #5
print(len({'a' : 1, 'b’: 2,’c': 3})) #3
class MyClass (object):
pass
print(len(MyClass())) #运行错误
如果想让内置函数len( )的实参可以是自定义类对象的实例对象,必须在自定义类对象中实现特殊方法__len__ (), 这样,调用内置函数Len()时,在其内部会自动调用实参所对应类对象的特殊方法__len__().之所以内置函数len()的实参可以是上述内置类对象的实例对象,是因为上述的内置类对象中都实现了特殊方法__len__()。
生成器
查看生成器对应的所有元素,有两种方式:
- 多次调用内置函数next(),每次调用都返回生成器的下一个元素,直到抛出异常StopIteration时表示没有更多元素了。
- 使用for-in语句对生成器进行迭代,这样就不需要关心异常StopIteration了。
生成器函数:
- 生成器中保存的并不是其对应的所有元素,而是如何推算出所有元素的算法。将生成器用于for-in语句时元素是在循环的过程中不断被推算出来的。将生成器作为内置函数next()的实参时,返回的下一个元素也是在调用函数时被推算出来的。因此,生成器是惰性推算的,也就是说,只有当用到生成器中的某个元素时,才会临时进行推算,而并不会提前推算出来。
- 如果需要创建一个元素个数较大的容器,就可以考虑使用生成器,从而节省大量的存储空间。
- 上面使用类似生成式的语法得到的生成器被称为生成器表达式。此外,当推算的算法比较复杂时,还可以使用生成器函数得到生成器。
- 生成器函数中通过关键字yield返回推算出的元素。生成器函数与普通函数的区别在于:当调用内置函数next()或使用for-in语句进行迭代时,执行完yield语句就会将生成器函数挂起,下次会从挂起的地方继续执行。
迭代器
1.可以用于for-in语句的对象被称为可迭代对象(Iterable)对象
2.可以调用内置函数isinstance()判断一个对象是否是可迭代对象。标准库模块collection中的类Iterable用于表示可迭代对象。
from collections import Iterable
print(isinstance([1, 2, 3], Iterable)) #True
print (isinstance( abc' , Iterable)) #True
print (isinstance((i * i for i in range(1, 7)), Iterable)) #True
1. 如果一个可迭代対象可以作カ内置函数next()的实参从而支持惰性推算,那么该対象被称为迭代器(Iterator) 对象。
2. 对于range、列表、元组、字符串、字典和集合等可迭代対象,都不可以作カ内置函数next(
)的实参,而生成器可以。所以,生成器是迭代器的一种。
3. 可以调用内置函数isinstance()判断一个对象是否是迭代器对象。标准库模块collections中的类Iterator用于表示迭代器对象。
4. 可以调用内置函数iter()把不支持惰性推算的可迭代对象转换为迭代器对象。
5. 如果一个对象同时实现了特殊方法__iter__ ()和__next__ (), 那么该对象也被称为迭代器对象。如果将该对象用于for-
in语句,for- in语句首先会调用特殊方法__iter__()返回一个可迭代对象,然后不断调用该可迭代对象的特殊方法__next__
()返回下一 次迭代的值直到遇到StopIteration时退出循环。