python中的生成器、迭代器、装饰器详解

一、装饰器

由于一个函数能实现一种功能,现在想要在不改变其代码的情况下,让这个函数进化一下,即能保持原来的功能,还能有新的"技能",怎么办?

现已经存在一个自定义的函数func1

def func1():
    print('hello,world!')

让func1进化一下:(继承func1之前的所有功能,而且还有新的‘技能’)

效果和下面定义的函数func2效果是一样的

def func2():
    func1() #调用func1,即可保持func1这一函数的所有的功能都被这个新的函数继承下来
    print('hello,boy!') #添加的新功能,相当于func1这一函数学到的新技能

但是,func2是一个新的函数,已经完全改头换面了,虽然有一部分是能实现func1的功能,但并不是func1的进化型,所以当我们还想用调用func1这个函数的调用方法调用的时候,并不能调用func2.

当函数不进行调用时候,可以将这个函数当成一个变量进行对待。所以,如果把func2这个函数的内存地址赋给func1,然后调用func1,那么就能用func1这个名字,调用func2这个函数,也就实现了func1这个函数的进化。

所以,如果定义如下一个可以实现上述功能的函数deco,deco这个函数就要完成以下的两件事情:

1.让func2这个函数的内存地址出现,即:定义func2这个函数

2.在其执行之后能够,拿到func2的内存地址,即把func2的内存地址作为返回值返回

def deco(func1):
    def func2():
        func1() #调用func1,即可保持func1这一函数的所有的功能都被这个新的函数继承下来
        print('hello,boy!') #添加的新功能,相当于func1这一函数学到的新技能
    return func2

完成上述deco函数的定义之后,当执行deco这个函数之后,其执行结果,就是func2的内存地址。

接下来,将这个内存地址赋值给func1这个变量之后,再对func1进行调用,就可以完成对函数func1的"进化"(即:在不改变func1的代码,还赋予了其新的功能)

func1 = deco(func1)
func1()

上述的过程可以用以下的代码进行实现:

def deco(func1):
    def func2():
        func1() #调用func1,即可保持func1这一函数的所有的功能都被这个新的函数继承下来
        print('hello,boy!') #添加的新功能,相当于func1这一函数学到的新技能
    return func2

def func1():
    print('hello,world!')

func1 = deco(func1)
func1()

其中,deco这个函数就是所谓的装饰器

(装饰器:在不改变源代码和调用方式的基础之上给函数增加新的功能)

将上述代码进行优化之后就有了下面的代码:

def deco(func1):
    def func2():
        func1() #调用func1,即可保持func1这一函数的所有的功能都被这个新的函数继承下来
        print('hello,boy!') #添加的新功能,相当于func1这一函数学到的新技能
    return func2

@deco #效果等同于func1=deco(func1)
def func1():
    print('hello,world!')

func1()

1.2含参数的装饰器:

def deco(func):
    def wrapper(username,password):
        if username == 'root' and password == 'root':
            func(username,password)  else:
            print('用户名或密码错误')
    return wrapper

@deco
def baidu_index(username,password):
    print('welcome to 百度')

baidu_index('root','root')

由于定义的函数baidu_index,必须要传递参数,所以装饰器内部定义的函数wrapper也需要定义形参,wrapper函数内部调用函数时,也需要有参数!!

1.3多层装饰器

将装饰器1看成一个整体,在这个装饰器上在添加一个装饰器2,就能实现..........

例如:

def deco1(deco):
    print('你好不好?')
    def deco(func):
        def func2():
            print('你不好!')
            func()
        return func2
    return deco

@deco1
def deco(func):
    def func2():
        print('你不好!')
        func()
    return func2

@deco
def func1():
    print('你好!')

func1()

二、迭代器:

1.什么是迭代?

1.迭代是一个重复的过程,即每一次重复为一次迭代,

2.并且每次迭代的结果都是下一次迭代的初始值

例如:

l = [1,2,3]
count=0
while count<len(l): #首先是重复动作,其次上一次的结果是下一次的初始值,因此,是迭代
  print(l[count])
  count+=1

2.什么是迭代器?为何要有迭代器?

对于序列类型:字符串、列表、元组,我们可以使用索引的方式迭代取出其包含的元素。但对于字典、集合、文件等类型是没有索引的,若还想取出其内部包含的元素,则必须找出一种不依赖于索引的迭代方式,这就是迭代器。

3.什么叫做迭代器对象?

obj有.__iter__和.__next__方法的叫做迭代器对象

总结:迭代器对象一定是可迭代对象,而可迭代对象不一定是迭代器对象

4.for的作用:

1.把可迭代对象变成迭代器对象

2.过滤错误信息

l1 = [1,2,3]
for i in l1: #iter(l1)
  print(i)

三、生成器

1.什么是生成器?

只要函数里有yield关键字,那么函数名()得到的结果就是生成器,生成器就是迭代器,并且不会执行函数内部代码

2.return和yield用法十分类似,但是也有区别,区别在于:return只能返回一个值,而yield可以返回多个值

3.生成器优点:

同一时间只存储一个值,节省内存空间

4.生成器的缺点:

只能向后取值,不能往前取值

def test():
  for i in range(100):
    yield i

res = test()

for k in res:
  print(k)

四、总结

迭代器

迭代是Python最强大的功能之一,是访问集合元素的一种方式;迭代器是一个可以记住遍历的位置的对象;迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束。迭代器只能往前不会后退;迭代器有两个基本的方法:iter()和next();字符串,列表或元组对象都可用于创建迭代器:

迭代器python实例

生成器

在 Python 中,使用了 yield 的函数被称为生成器;跟普通函数不同的是,生成器是一个返回迭代器的函数,只能用于迭代操作,更简单点理解生成器就是一个迭代器;在调用生成器运行的过程中,每次遇到 yield 时函数会暂停并保存当前所有的运行信息,返回 yield 的值, 并在下一次执行 next() 方法时从当前位置继续运行;调用一个生成器函数,返回的是一个迭代器对象。

使用生成器生成斐波那些数列

装饰器

装饰器:在不改变原函数的基础上,对函数执行前后进行自定义操作。把目标函数作为参数传给装饰器函数,装饰器函数执行过程中,执行目标函数,达到在目标函数运行前后进行自定义操作的目的。

应用场景:如记录函数运行时间;flask里的路由、before_request;django中的缓存、用户登录等。

使用装饰器记录函数运行时间

装饰器在实现的时候,被装饰后的函数其实已经是另外一个函数了(函数名等函数属性会发生改变),为了不影响,Python的functools包中提供了一个叫wraps的装饰器来消除这样的副作用。写一个装饰器的时候,最好在实现之前加上functools的wrap,它能保留原有函数的名称和文档字符串。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • 老生常谈Python之装饰器、迭代器和生成器

    在学习python的时候,三大"名器"对没有其他语言编程经验的人来说,应该算是一个小难点,本次博客就博主自己对装饰器.迭代器和生成器理解进行解释. 为什么要使用装饰器 什么是装饰器?"装饰"从字面意思来谁就是对特定的建筑物内按照一定的思路和风格进行美化的一种行为,所谓"器"就是工具,对于python来说装饰器就是能够在不修改原始的代码情况下给其添加新的功能,比如一款软件上线之后,我们需要在不修改源代码和不修改被调用的方式的情况下还能为期添加新的功

  • python三大器之迭代器、生成器、装饰器

    目录 迭代器 生成器 装饰器(非常实用!) 迭代器 聊迭代器前我们要先清楚迭代的概念:通常来讲从一个对象中依次取出数据,这个过程叫做遍历,这个手段称为迭代(重复执行某一段代码块,并将每一次迭代得到的结果作为下一次迭代的初始值).可迭代对象(iterable):是指该对象可以被用于for…in…循环,例如:集合,列表,元祖,字典,字符串,迭代器等. 在python中如果一个对象实现了 __iter__方法,我们就称之为可迭代对象,可以查看set\list\tuple…等源码内部均实现了__iter

  • python的迭代器,生成器和装饰器你了解吗

    python 迭代器与生成器,装饰器 迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束. 迭代器有两个基本的方法:iter() 和 next(). 字符串,列表或元组对象都可用于创建迭代器: list1=[1,2,3] s=iter(list1) # 创建迭代器对象 print(next(s)) # 输出迭代器的下一个元素 print(next(s)) print(next(s)) 直到计算出最后一个元素,没有更多的元素时,抛出StopIteration的错误 迭代器对象可以使用

  • python中的迭代器,生成器与装饰器详解

    目录 迭代器 生成器 装饰器 总结 迭代器 每一个可迭代类内部都要实现__iter__()方法,返回一个迭代类对象,迭代类对象则定义了这个可迭代类如何迭代. for循环调用list本质上是是调用了list的迭代器进行迭代. # 对list进行for循环本质上是调用了list的迭代器 list = [1,2,3,4] # for 循环调用 for elem in list: print(elem) # 迭代器调用 list_iter = list.__iter__() while True: tr

  • 详解python中的生成器、迭代器、闭包、装饰器

    迭代是访问集合元素的一种方式.迭代器是一个可以记住遍历的位置的对象.迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不会后退. 1|1可迭代对象 以直接作用于 for 循环的数据类型有以下几种: 一类是集合数据类型,如 list . tuple . dict . set . str 等: 一类是 generator ,包括生成器和带 yield 的generator function. 这些可以直接作用于 for 循环的对象统称为可迭代对象: Iterable .

  • python中函数总结之装饰器闭包详解

    1.前言 函数也是一个对象,从而可以增加属性,使用句点来表示属性. 如果内部函数的定义包含了在外部函数中定义的对象的引用(外部对象可以是在外部函数之外),那么内部函数被称之为闭包. 2.装饰器 装饰器就是包装原来的函数,从而在不需要修改原来代码的基础之上,可以做更多的事情. 装饰器语法如下: @deco2 @deco1 def func(arg1,arg2...): pass 这个表示了有两个装饰器的函数,那么表示的含义为:func = deco2(deco1(func)) 无参装饰器语法如下:

  • Python 3.7新功能之dataclass装饰器详解

    前言 Python 3.7 将于今年夏天发布,Python 3.7 中将会有许多新东西: 各种字符集的改进 对注释的推迟评估 以及对dataclass的支持 最激动人心的新功能之一是 dataclass 装饰器. 什么是 Data Class 大多数 Python 开发人员编写过很多像下面这样的类: class MyClass: def __init__(self, var_a, var_b): self.var_a = var_a self.var_b = var_b dataclass 可以

  • python高级语法之闭包和装饰器详解

    一.闭包 闭包的形成条件: 1.函数嵌套. 2.内部函数使用了外部函数的变量或者参数. 3.外部函数返回了使用外 部变量的内部函数. 二.一个简单的例子 def func_out(num1): def inner(num2): res = num1 + num2 print(res) return inner # a = func_out(10)(10) a = func_out(10) a(10) 闭包修改外部函数的变量: 在闭包内修改外部函数的变量需要使用nonlocal关键字 def fu

  • python 装饰器详解与应用范例

    什么是装饰器 从字面意思上来看,装饰器是用来装饰其他东西的工具.在python中装饰器分为函数装饰器和类装饰器. 简而言之,函数装饰器是用来装饰函数的装饰器,其主要目的是增加目标函数的功能,类装饰器也就是装饰类的装饰器,增加类的功能. 函数装饰器 装饰器本质是嵌套函数 下面是一个简单的装饰器 # fun1为装饰器名称,function指的是被装饰的函数 def fun1(function): def fun2(): print("开始了!") function() # 执行被装饰的函数

  • Python学习之装饰器与类的装饰器详解

    目录 装饰器 装饰器的定义 装饰器的用法 类中的装饰器 类的装饰器-classmethod 类的装饰器-staticmethod 类的装饰器-property 通过学习装饰器可以让我们更好更灵活的使用函数,通过学会使用装饰器还可以让我们的代码更加优雅. 在我们的实际工作中,很多场景都会用到装饰器,比如记录一些日志.或者屏蔽一些不太合法的程序执行从而使我们的代码更加安全. 装饰器 什么是装饰器?虽然对这个次感到陌生,但是完全不需要担心. 首先,装饰器也是一种函数:只不过装饰器可以接收 函数 作为参

  • 深入理解python中的闭包和装饰器

    python中的闭包从表现形式上定义(解释)为:如果在一个内部函数里,对在外部作用域(但不是在全局作用域)的变量进行引用,那么内部函数就被认为是闭包(closure). 以下说明主要针对 python2.7,其他版本可能存在差异. 也许直接看定义并不太能明白,下面我们先来看一下什么叫做内部函数: def wai_hanshu(canshu_1): def nei_hanshu(canshu_2): # 我在函数内部有定义了一个函数 return canshu_1*canshu_2 return

  • python中property和setter装饰器用法

    作用:调用方法改为调用对象, 比如 : p.set_name() 改为 p.set_name 区别:前者改变get方法,后者改变set方法 效果图: 代码: class Person: def __init__(self,name): self._name = name def get_name(self): return self._name def set_name(self,name): self._name = name p = Person('小黑') print(p.get_name

  • python基础之装饰器详解

    一.前言 装饰器:本质就是函数,功能是为其他函数添加附加功能 原则: 1.不修改被修饰函数的源代码 2.不修改被修饰函数的调用方式 装饰器 = 高阶函数 + 函数嵌套 + 闭包 二.高阶函数 高阶函数定义: 1.函数接收的参数是一个函数 2.函数的返回值是一个函数名 3.满足上述条件任意一个,都可以称为高阶函数 test 函数是高阶函数,接受了一个foo 作为参数 import time def foo(): time.sleep(3) print("sleep 3s") def te

  • 对python中return与yield的区别详解

    首先比较下return 与 yield的区别: return:在程序函数中返回某个值,返回之后函数不在继续执行,彻底结束. yield: 带有yield的函数是一个迭代器,函数返回某个值时,会停留在某个位置,返回函数值后,会在前面停留的位置继续执行,直到程序结束 首先,如果你还没有对yield有个初步分认识,那么你先把yield看做"return",这个是直观的,它首先是个return,普通的return是什么意思,就是在程序中返回某个值,返回之后程序就不再往下运行了.看做return

随机推荐