"""
今日内容:
1、带参装饰器及warps
2、迭代器
3、生成器
""" """
# 一、带参装饰器及warps系统装饰器
# 1、为什么需要带参装饰器?
-- 昨天的内容已经讲了装饰器,但是讲的都是增加的函数不需要外界传人参数,如果装饰器也需要外界传入参数呢?
-- 此时就需要带参装饰器 # 2、装饰器的推导过程
-- 如果添加的功能也需要外界传入参数时,参数应该如何通过谁进行传递呢?
-- 如果通过inner传入参数,因为inner中为可变长参数,所以会直接将参数传递给原函数,此方法不可行
-- 如果通过 outter 进行传入,那么需要在func旁再增加参数,在调用outter时就需要将参数传递进去,此时就不能使用语法糖,破坏装饰器的整体结构
-- 所以,最后的办法是根据闭包的方法将整个装饰器作为闭包,在外界增加一层函数,最后将整个装饰器作为返回值进行返回,就可以为装饰器传递参数了 def wrap(info):
def outer(func):
# info = 0
def inner(*args, **kwargs):
print('新:拓展的新功能,可能也需要外界的参数%s' % info)
res = func(*args, **kwargs)
return res
return inner
return outer -- 此时,整个 warp 就是一个带参装饰器,其语法糖为 @wrap("新功能需要的参数") @wrap('外部参数')
def fn(): pass # 3、什么是wraps装饰器?
-- 系统自带的装饰器,用来在用户查询被装饰函数的doc文档时,将被装饰函数的文档注释打印给用户
-- 因为当我们使用装饰器为某个功能进行装饰时,最后的调用过程虽然看着还是原函数,但是我们已经为原函数进行了重新复制,此时再使用查询原函数的doc等内置功能时,访问的实际是装饰器
-- @warps装饰器的作用就是将原函数的doc等内置功能传递给用户,彻底伪装好装饰器 # 4、如何使用warp装饰器?
from functools import wraps
def outer(func):
@wraps(func)
def inner(*args, **kwargs): res = func(*args, **kwargs)
return res
return inner @outer
def fn(): pass """ """
# 二、迭代器
# 1、什么是迭代器?
-- 迭代器是一种可以不依赖于索引取值的容器 # 2、迭代器的优缺点:
-- 迭代器优点:可以不用依赖索引取值
-- 迭代器缺点:只能从前往后依次取值 # 3、迭代器对象与可迭代对象
-- 可迭代对象:具有__iter__()方法的对象就叫做可迭代对象,可迭代对象可以通过__iter__()方法转换成迭代器对象 ls = [4, 1, 5, 2, 3]
res = ls.__iter__() # => 可迭代对象
print(res) # <list_iterator object at 0x000002732B0C7470> -- 迭代器对象:具有__next__()方法的对象叫做迭代器对象,迭代器对象可以通过__next__()方法进行取值,一次只能取出一个值 with open('1.txt', 'rb') as f:
res = f.__next__() # 文件中的第一行内容
print(res)
res = f.__next__() # 文件中的第二行内容
print(res) # 4、for循环迭代器
-- 迭代器取值时的问题
-- 在我们从迭代器对象中取值时,可以使用__next__()从迭代器中取值,但是,每使用一次__next__()方法才会取出一个值,那么怎么把迭代器中的值全部取出来呢?
-- 在使用__next__()进行取值时,如果取出所有的值后还有__next__()方法,就会抛出异常 -- for循环迭代器
-- 从迭代器中取值,可以依次取出迭代器中所有的值,
-- for循环中封装了异常处理机制,会自动处理取去所有值后抛出的异常
-- for循环会将 in 后面的可迭代对象或迭代器对象直接添加.__iter__()方法 -- for循环迭代器的工作原理:
for v in obj: pass
-- 获取obj.__iter__()的结果,就是得到要操作的迭代器对象
-- 迭代器对象通过__next__()方法进行取值,依次将当前循环的取值结果赋值给v
-- 当取值抛异常,自动处理StopIteration异常结束取值循环 # 5、枚举器 (enumerate)
-- 枚举器就是给迭代器中的取值结果加索引的,当使用枚举器进行取值时,每次取值时会自动为取的值加上取值索引。
s = 'abc'
for v in enumerate(s):
print(v) # (0 'a') | (1 'b') | (2 'c') """ """
# 四、生成器
# 1、什么是生成器?
-- 生成器就是可以自定义的迭代器
# 2、如何自定义生成器?
-- 我们可以使用函数进行声明,再使用 yeild 返回值,这样在使用函数名()时就不是调用函数,而是生成迭代器对象
-- 在有yield关键字时,再使用return关键字,return关键字不起任何作用
-- 使用yeild关键字生成迭代器时,当使用__next__()进行取值时,每次只会返回一个yeild后的值,并卡在原地,不会执行后面的代码,当再次遇到__next__()时才会继续向下执行
def fn():
yield 1
yield 3
yield 5
obj = fn()
obj.__next__() # 从开始往下执行,遇到第一个yield停止,拿到yield的返回值
obj.__next__() # 从上一次停止的yield往下执行,在再遇到的yield时停止,拿到当前停止的yield的返回值
... # 以此类推,直到无法获得下一个yield,抛StopIteration异常 -- 我们自定义的生成器可以使用for循环直接取值。 # 3、生成器案例 # 案例一:创建生成器,从其取值,依次得到1! 2! 3! ...,可无限取值生成器。
def jiecheng():
ji = 1
count = 1
while True:
ji *= count
yield ji
count += 1 obj = jiecheng()
print(obj.__next__())
print(obj.__next__())
print(obj.__next__()) # 可以无限取 # 案例二:
# 自定义结束条件生成器
def jiecheng_num(num):
ji = 1
for i in range(1, num + 1):
ji *= i
yield ji
# ... obj = jiecheng_num(3)
print(obj.__next__())
print(obj.__next__())
print(obj.__next__())
print(obj.__next__()) # 有异常了 for v in jiecheng_num(5):
print(v) # 会自动处理异常停止 # 案例三:
# 自制的range函数
def my_range(num): # => [0, 1, 2, ..., num - 1]
count = 0
while count < num:
yield count
count += 1 for v in my_range(10):
print(v, end=' ') print(list(my_range(10)))
"""