0. 引子
1. 函数名的运用
- 函数名是一个变量, 但它是一个特殊的变量, 与括号配合可以执行函数的变量
2. 函数名的内存地址
def func():
print("呵呵")
print(func)
结果: <function func at 0x1101e4ea0>
3. 函数名可以赋值给其他变量
def func():
print("呵呵")
print(func)
a = func # 把函数当成一个值赋值给另一个变量
a() # 函数调用 func()
4. 函数名可以当做容器类的元素
def func1():
print("呵呵")
def func2():
print("呵呵")
def func3():
print("呵呵")
def func4():
print("呵呵")
lst = [func1, func2, func3]
for i in lst:
i()
5. 函数名可以当做函数的参数
def func():
print("吃了么")
def func2(fn):
print("我是func2")
fn() # 执行传递过来的fn
print("我是func2")
func2(func) # 把函数func当成参数传递给func2的参数fn.
6. 函数名可以作为函数的返回值
def func_1():
print("这里是函数1")
def func_2():
print("这里是函数2")
print("这里是函数1")
return func_2
fn = func_1()
# 执行函数1. 函数1返回的是函数2, 这时fn指向的就是上面函数2
fn() # 执行func_2函数
7. 闭包
- 什么是闭包? 闭包就是内层函数, 对外层函数(非全局)的变量的引用. 叫闭包
def func1():
name = "alex"
def func2():
print(name)
# 闭包
func2()
func1()
# 结果: alex
8. 检测闭包
- 我们可以使用__closure__ 来检测函数是否是闭包. 使用函数名.__closure__返回cell,就是闭包. 返回None就不是闭包
def func1():
name = "alex"
def func2():
print(name)
func2()
print(func2.__closure__)
func1()
结果:
alex
(<cell at 0x0000020077EFC378: str object at 0x00000200674DC340>,)
1. 迭代器
1.1 迭代器类的定义
- 当类中定义了__iter__和__next__ 两个方法
- __iter__需要返回对象本身,既self
- __next__返回下一个数据,如果没有数据,就会抛异常
class T2:
def __init__(self):
self.num = 0
def __iter__(self):
return self
def __next__(self):
self.num += 1
if self.num == 3:
raise StopIteration()
return self.num
obj = T2()
v1 = next(obj) # obj.__next__
obj2 = T2()
for item in obj2:
print(item) # for 循环在内部循环时,先执行__iter__获取迭代器对象,然后执行__next__,如果没有值了,抛异常
1.1 可迭代对象定义
- 如果一个类中有__iter__且返回一个迭代器,那么我们称这个类创建的对象,叫可迭代对象
class T3:
def __iter__(self):
return iter([1, 2, 3])
obj3 = T3()
for item in obj3: # 执行其__iter__ 获取迭代器
print(item)
- 熟知的可迭代对象有哪些:
str list tuple dic set
- 那为什么我们称他们为可迭代对象呢?
- 因为他们都遵循了可迭代协议,那什么又是可迭代协议呢.首先我们先看一段代码:
- 正确的代码:
s = 'abc'
for i in s:
print(i)
结果:
a
b
c
错误的代码:
for i in 123:
print(i)
结果
Traceback (most recent call last):
File "D:/python_object/二分法.py", line 62, in <module>
for i in 123:
TypeError: 'int' object is not iterable
- 注意看报错信息,报错信息中有这样一句话: ‘int’ object is not iterable 翻译过来就是整数类型对象是不可迭代的.
- iterable表示可迭代的.表示可迭代协议 那么如何进行验证你的数据类型是否符合可迭代协议.我们可以通过dir函数来查看,类中定义好的所有方法
a = 'abc'
print(dir(a)) # dir查看对象的方法和函数
# 在打印结果中寻找__iter__ 如果存在就表示当前的这个类型是个可迭代对象
1.2 __iter__ 的校验
# 列表
lst = [1,2]
print(dir(lst))
# 元祖
tuple = (1,2)
print(dir(tuple))
# 字典
dic = {'a':1,'b':2}
print(dir(dic))
# 集合
se = {1,2,3,4,4}
print(dir(se))
- 是不是发现以上都有__iter__,并且还可以for循环啊,
- 其实也可以这么说,可以for循环的就有__iter__方法,包括range
1.3 for循环机制
- for循环是不是也可以,并且还不报错啊,其实上边就是for的机制,
- 注意: 迭代器不能反复,只能向下执行,并且是一次性的.获取过了就不能在获取了
s = "我爱北京天安⻔"
c = s.__iter__() # 获取迭代器
print(c.__next__()) # 使⽤迭代器进⾏迭代. 获取⼀个元素 我
print(c.__next__()) # 爱
print(c.__next__()) # 北
print(c.__next__()) # 京
print(c.__next__()) # 天
print(c.__next__()) # 安
print(c.__next__()) # ⻔
print(c.__next__()) # StopIteration
1.4 总结
- Iterable: 可迭代对象. 内部包含__iter__()函数
- Iterator: 迭代器. 内部包含__iter__() 同时包含__next__().
- 迭代器的特点:
1. 节省内存.
2. 惰性机制m
3. 不能反复, 只能向下执行.
- 我们可以把要迭代的内容当成子弹. 然后呢. 获取到迭代器__iter__(), 就把子弹都装在弹夹中. 然后发射就是__next__()把每一个子弹(元素)打出来.
- 也就是说, for循环的时候.一开始的时候是__iter__()来获取迭代器.
- 后面每次获取元素都是通过next()来完成的. 当程序遇到 StopIteration将结束循环.