赞
踩
1. def spam():
2. yield”first”
3. yield”second”
4. yield”third”
5. for x in spam:
6. Print(x)
7. -----结果如下-----
8. first
9. second
10. third
1 我们创建列表的时候,受到内存限制,容量肯定是有限的,而且不可能全部给他一次枚举出来。Python常用的列表生成式有一个致命的缺点就是定义即生成,非常的浪费空间和效率。
2 若列表元素可以按照某种算法推算出来,那么在循环的过程中不断推算中后续的元素,这样就不必创建完整的list,从而节省大量的空间。在python中,这种一边循环一边计算的机制,称为生成器:generator。与字典和集合相反,生成器是典型的时间换空间的例子。
3 创建一个generator,最简单的方法就是改造列表生成式,还有就是生成器函数,通过def定义,然后使用yield来支持迭代器协议,比迭代器写起来更简单。
4 进行函数调用的时候,返回一个生成器对象。在使用next()调用的时候,遇到yield就返回,记录此时的函数调用位置,下次调用next()时,从断点处开始。
5 我们完全可以像使用迭代器一样使用generator,当然除了定义,定义一个迭代器,需要分别实现iter()和next()方法,但generator只需要yield就可以。
虽然生成器和迭代都可以用于for循环的遍历,但它们之间有明显的不同之处:
装饰器本质上是还是一个函数,它可以让已有函数不做任何改动的情况下增加功能。
装饰器允许通过将现有函数传递给装饰器,从而向现有函数添加一些额外的功能,该装饰器将执行现有函数的功能和添加的额外功能。
装饰器两大特性:
Eg:写一个记录函数执行时间的装饰器【面试常考】:
1. from time import time
2.
3. def record_time(func):
4. def inner(*args, **kwargs): # 这里的*args和**kwargs用来接收被装饰的函数的参数
5. """函数注释:用来修饰被装饰函数的内嵌函数"""
6. start = time()
7. result = func(*args, **kwargs) # result用来接受被装饰的函数的返回值
8. end = time()
9. print(f"被装饰的函数{func.__name__}的执行时间为:{end - start}")
10. return result + '~~~~~~' # 对被装饰的函数的返回值进行修饰
11.
12. return inner
上面定义的record_time函数即为一个装饰器,内嵌函数inner+外部函数参数fun即构成了闭包。
1. def countdown(n):
2. """函数注解:这是一个倒计时函数"""
3. print("开始倒计时")
4. while n > 0:
5. n -= 1
6. print("倒计时结束")
7. return "我执行完啦"
8.
9. # 使用方法1
10. countdown = record_time(countdown) # countdown是 record_time 函数中返回的 inner 函数
11. res = countdown(1000000) # 执行 inner 函数,将 inner 函数的返回值给 res
12. print(res)
1. @record_time # 对人类更友好的语法糖写法
2. def countdown(n):
3. """函数注解:这是一个倒计时函数"""
4. print("开始倒计时")
5. while n > 0:
6. n -= 1
7. print("倒计时结束")
8. return "我执行完啦"
9.
10. res = countdown(10000000) # 被装饰完的 countdown 虽然看起来和调用方式都是 countdown,其实是它的本质是一个 inner 函数
11. print(res)
语法1和语法2的原理是一样的,写法2是语法糖的写法,语法糖是指那些没有给计算机语言添加新功能或特性的语法结构。
装饰器本质上是一个python函数,它可以让其他函数在不需要做任何代码变动的前提下增加额外的功能。装饰器的返回值也是一个函数对象,它适用于有切面需要的场景。比如:插入日志、性能测试、事务处理、缓存、权限校验等场景,装饰器是解决这类问题的绝佳设计,有了装饰器,我们就可以抽离出大量与函数功能本身无关的雷同代码并继续重用。
1. def first(fun):
2. def second():
3. print('start')
4. fun()
5. print('end')
6. print fun.__name__
7. return second
8.
9. def man():
10. print('i am a man()')
11.
12. f = first(man)
13. f()
等价于
1. 1. def first(func): 2. 2. def second(): 3. 3. print('start') 4. 4. func() 5. 5. print('end') 6. 6. print (func.__name__) 7. 7. return second 8. 8. 9. 9. @first 10. 10. def man(): 11. 11. print('i am a man()') 12. 12. 13. 13. man() 14. ------ 结果 ------ 15. Start 16. I am a man() 17. End 18. man
上面的这段代码和之前的代码的作用一模一样。区别在于之前的代码直接“明目张胆”的使用first函数去封装函数,而上面这个用了【语法糖】来封装man函数。在上述代码中【@first】在man函数的上面,表示对man函数使用first装饰器。【@】是装饰器的语法,【first】是装饰器的名称。
函数可以作为参数传递的语言,可以使用装饰器。
语言区分作用域,是为了复用变量名。引入作用域,相当于给变量划分了各自的‘隔离区’,在不同的‘隔离区’里,查找变量变得容易。Python中,变量的作用域取决于其定义的位置。当python遇到一个变量的话它会按照这个顺序进行搜索(由内向外):
局部作用域(Local)–>嵌套作用域(Enclosing locals)–>全局作用域(Global)–>内置作用域(Built-in)
闭包指的是在函数内部定义了另一个函数,并返回了这个作为函数的对象,同时还保存了外层函数的状态信息。这个嵌套函数可以引用外层函数的变量和参数,而且外层函数返回的是这个嵌套函数的引用。这种在函数内部定义函数并返回的方式称为闭包。闭包延伸了嵌套函数的作用域,其中主要指在函数定义中引用但不在函数定义体中定义的非全局变量。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。