首页 > 编程语言 >Python高级之生成器

Python高级之生成器

时间:2024-05-09 15:25:42浏览次数:30  
标签:Python res 生成器 高级 yield next start print

【一】什么是生成器

  • Python中的生成器是一种特殊的迭代器
  • 可以在需要时生成数据,而不必提前从内存中生成并存储整个数据集
  • 通过生成器,可以逐个生成序列中的元素,而无需一次性生成整个序列

【二】生成器的创建方式

【1】列表推导式

# 列表生成式生成列表
num_list = [i for i in range(10)]
print(num_list)
# [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]

# 将列表的中括号变成小括号,变成元祖,生成的是一个生成器对象
num_list = (i for i in range(10))
print(num_list)
# <generator object <genexpr> at 0x000001D26E884270>

#可以利用for循环来打印生成器的每一个元素
num_list = (i for i in range(5))
for i in num_list:
    print(i)
# 0
# 1
# 2
# 3
# 4

【2】yield关键字

(1)yield关键字介绍

  • 使用yield关键字定义一个生成器函数时,生成器函数中的yield语句会暂停函数执行并返回一个值,下一次调用该函数时会继续执行并返回下一个值
def add():
    yield 1
    yield 2
    yield 3


res = add()
print(next(res))  # 1
print(next(res))  # 2
print(next(res))  # 3


def add(i):
    while True:
        yield i


res = add(1)
print(next(res))  # 1
res = add(2)
print(next(res))  # 2
res = add(3)
print(next(res))  # 3

(2)yield关键字实例使用

def eater():
    print('开始吃饭 ovo ')
    while True:
        food = yield
        print(f'得到的食物是 :>>>> {food}, 开始吃饭喽 :>>>> {food}')


eater = eater()
print(eater.__next__())
print(eater.__next__())
# 开始吃饭 ovo
# None   yield中没有返回值
# 得到的食物是 :>>>> None, 开始吃饭喽 :>>>> None
# None   在生成器内部走了一圈回来又到了yield,但是yield没有返回值

#需要给生成器传值
eater = eater()
# 需要事先”初始化”一次,让函数挂起在food=yield,等待调用eater.send()方法为其传值
# eater.send(None)
# 等同于
next(eater)
eater.send('土豆丝')
eater.send('酸菜鱼')
# 开始吃饭 ovo
# 得到的食物是 :>>>> 土豆丝, 开始吃饭喽 :>>>> 土豆丝
# 得到的食物是 :>>>> 酸菜鱼, 开始吃饭喽 :>>>> 酸菜鱼

【三】装饰器 + 生成器

def outer(func):
    # func 我的生成器函数
    def inner(*args, **kwargs):
        # function得到的生成器对象
        function = func(*args, **kwargs)
        # 调用自己生成器向下走
        next(function)
        # 走回来的返回值返回出去
        return function

    return inner


@outer
def eater():
    print('开始吃饭 ovo ')
    while True:
        food = yield
        print(f'得到的食物是 :>>>> {food}, 开始吃饭喽 :>>>> {food}')


eater = eater()
eater.send('土豆丝')
eater.send('酸菜鱼')
# 开始吃饭 ovo 
# 得到的食物是 :>>>> 土豆丝, 开始吃饭喽 :>>>> 土豆丝
# 得到的食物是 :>>>> 酸菜鱼, 开始吃饭喽 :>>>> 酸菜鱼

【四】生成器内部修改可变数据类型

def outer(func):
    # func 我的生成器函数
    def inner(*args, **kwargs):
        # g 得到的生成器对象
        function = func(*args, **kwargs)
        # 调用自己生成器向下走
        next(function)
        # 走回来的返回值返回出去
        return function

    return inner


@outer
def eater():
    print('开始吃饭 ovo ')
    food_list = [] #定于一个空字典
    while True:
        food = yield
        food_list.append(food) #将获取的元素一个一个加到列表中
        print(f'得到的食物是 :>>>> {food}, 开始吃饭喽 :>>>> {food}')
        print(food_list)


eater = eater()
eater.send('土豆丝')
eater.send('酸菜鱼')
# 开始吃饭 ovo 
# 得到的食物是 :>>>> 土豆丝, 开始吃饭喽 :>>>> 土豆丝
# ['土豆丝']
# 得到的食物是 :>>>> 酸菜鱼, 开始吃饭喽 :>>>> 酸菜鱼
# ['土豆丝', '酸菜鱼']

【五】yield+next详解

  • 若函数体包含yield关键字,再调用函数,并不会执行函数体代码,得到的返回值即生成器对象
def my_range(start, stop, step=1):
    print('start...')
    while start < stop:
        yield start
        start += step
    print('end...')


res = my_range(0, 3)
print(g) # <generator object my_range at 0x0000019958788430>
  • 生成器内置有__iter__和__next__方法,所以生成器本身就是一个迭代器
def my_range(start, stop, step):
    print('start...')
    while start < stop:
        yield start
        start += step
    print('end...')


res = my_range(0, 10, 2)
print(res)  # <generator object my_range at 0x00000171824A4270>
print(res.__iter__)  # <method-wrapper '__iter__' of generator object at 0x0000012CE8CA5BD0>
print(res.__next__)  # <method-wrapper '__next__' of generator object at 0x0000012CE8CA5BD0>
print(res.__next__())
print(res.__next__())
print(next(res))
print(next(res))
print(next(res))
print(next(res))
# start...
# 0
# 2
# 4
# 6
# 8
# end...
# Traceback (most recent call last):
#   File "D:\Python\pythonProject\pythonProject1\demo7.py", line 383, in <module>
#     print(next(res))
# StopIteration

#在函数内部加上异常捕获
def init_iter(func):
    def inner(*args, **kwargs):
        try:
            generator = func(*args, **kwargs)
            next(generator)
            return generator
        except StopIteration:
            pass

    return inner


def range_(start, stop, step):
    while start < stop:
        yield start
        start += step


# for 循环内部做了异常捕获
for i in range_(0, 5, 1):
    print(i)


def my_range(start, stop, step):
    def range_(start, stop, step):
        while start < stop:
            yield start
            start += step

    res = range_(start, stop, step)
    while True:
        try:
            print(res.__next__())
        except StopIteration:
            break


res = my_range(start=1, stop=5, step=1)											

【六】生成器的特点

  • 节约内存
  • 迭代到下一次的调用时,所使用的参数都是第一次所保留下的
  • 在整个所有函数调用的参数都是第一次所调用时保留的,而不是新创建的

标签:Python,res,生成器,高级,yield,next,start,print
From: https://www.cnblogs.com/ligo6/p/18180003

相关文章

  • Python高级之迭代器
    【一】迭代器介绍迭代器就是迭代取值的工具,而迭代是重复反馈过程的活动其目的通常是为了逼近所需的目标或结果,而每一次迭代得到的结果会作为下一次迭代的初始值#只会重复让你输入信息,并不是迭代过程whileTrue:msg=input("请输入信息:").strip()print(msg)#下......
  • Python高级之【补充】算法
    【一】二分法【1】介绍二分法也称为折半法,是一种在有序数组中查找特定元素的搜索算法【2】思路首先,从数组的中间元素开始搜索,如果该元素正好是目标元素,则搜索过程结束,否则执行下一步如果目标元素大于/小于中间元素,则在数组大于/小于中间元素的那一半区域查找,然后重复步骤......
  • Python高级之常用的内置函数
    【一】什么是内置函数内置函数就是Python给你提供的,拿来直接用的函数目前共有68个内置函数Built-inFunctionsAabs()aiter()all()any()anext()ascii()Bbin()bool()breakpoint()bytearray()bytes()Ccallable()chr()classmethod()compile()complex()Ddelatt......
  • Python高级之模块与包
    【一】模块介绍【1】什么是模块在Python中,一个py文件就是一个模块,文件名为xxx.py模块名则是xxx,导入模块可以引用模块中已经写好的功能使用模块既保证了代码的重用性,又增强了程序的结构性和可维护性另外除了自定义模块外,我们还可以导入使用内置或第三方模块提供的现成功能......
  • 《最新出炉》系列入门篇-Python+Playwright自动化测试-45-鼠标操作-下篇
    1.简介鼠标为我们使用电脑提供了很多方便,我们看到的东西就可以将鼠标移动过去进行点击就可以打开或者访问内容,当页面内容过长时,我们也可以使用鼠标滚轮来实现对整个页面内容的查看,其实playwright也有鼠标操作的方法。上一篇文章中已经讲解过鼠标的部分操作了,今天宏哥在这里将剩下......
  • Python高级之函数参数进阶Optional
    【一】引言在Python3.5版本后引入的typing模块为Python的静态类型注解提供了支持。这个模块在增强代码可读性和维护性方面提供了帮助。本文将深入探讨typing模块,介绍其基本概念、常用类型注解以及使用示例,以帮助读者更全面地了解和应用静态类型注解。【二】基本类型注解【......
  • Python高级之名称空间和作用域
    【一】名称空间【1】什么是名称空间名称空间就是存放函数名与函数值对应关系的地方内存空间就是申请一块内存空间,然后将函数值放到内存空间里再将变量名和变量值绑定存到名称空间里程序执行期间最多会存在三种名称空间【2】内置名称空间会跟着python解释器的启动而生成,......
  • Python高级之匿名函数
    【一】匿名函数的定义在Python里有两类函数:用def关键词定义的正规函数用lambda关键词定义的匿名函数lambda参数:表达式lambda:定义匿名函数的关键词。函数参数它们可以是位置参数、默认参数、关键字参数表达式,输入函数参数,输出一些值,表达式本身结果就是返回......
  • Python高级之函数对象与闭包函数
    【一】函数对象函数对象是指函数可以被当成数据来处理,python中一切皆为对象【1】函数可以被引用defadd(a,b):returna+bres=add(3,4)print(res)#7【2】函数作为容器类型的元素defadd(a,b):returna+bnum_list=[add,1]res=num_list[0......
  • Python高级之函数的参数
    【一】形参和实参函数的参数分为形参和实参,形参就是定义在函数名后面括号里的参数(用来接收外部传来的值),实参就是调用函数时,括号里传进去的值(值可以是常量、变量、表达式)defadd(x,y):returnx+y#实参是常量print(add(3,4))#输出7#实参是变量x=3y=4prin......