开发者

python生成器函数yield与yield from使用

目录
  • 一、为什么叫生成器函数?
    • 1、yield在生成器函数中使用
  • 二、怎么使用generator类型的数据?
    • 1、什么是迭代器?
    • 2、generator生成器里可以放什么类型的数据?
    • 3、怎么使用generator类型的数据?
    • 3、generator和其他类型的互相转换
  • 三、yield from使用
    • 1、如果解读如下代码段?
    • 2、yield from后面要求跟什么?
    • 3、yield后面跟什么?
    • 4、yield from和yield的相同和不同点是什么
    • 5、解读如下代码
  • 总结

    一、为什么叫生成器函数?

    因为它们产生generate一系列值而得名。在生成器函数中,使用"yield"语句生成一个值并将其返回调用值,而不是通过普通的"return"语句一次性返回所有值。

    因为生成器函数只有在请求下一个值时,才会进行计算,并且在生产值时保留函数的状态。所以,处理大量的数据时,不会占用很大的内存。他们不会一次性计算并存储所有的值,所以主要特点是『惰性计算』。

    • 好处:高效地使用内存和计算资源。
    • 最大特点:惰性计算。(用的时候,才去计算。)
    • 保留状态:保留之前的值。

     如下代码,yiled调用生成器,返回的是一个generator类型的值。

    python生成器函数yield与yield from使用

    1、yield在生成器函数中使用

    生成器是一种特殊的函数。

    生成器函数中的 "yield" 关键词用于产生值并将其返回给调用者,同时保留函数的当前状态。

    每次调用生成器函数时,它会从上次 "yield" 语句的位置恢复执行,并一直执行到下一个 "yield" 语句。这样,生成器函数可以逐步生成值,而不是一次性生成所有值,从而节省内存并提高性能。

    如下代码,是一个生成器函数,它会生成一系列整数值:1,2,3。

    def nested_generator():
        yield 1
        yield 2
        yield 3

    二、怎么使用generator类型的数据?

    生成器generator是一种特殊类型的迭代器。

    1、什么是迭代器?

    迭代器Iterator,用来一种遍历或迭代集合元素的对象。可以按顺序访问集合的每个元素,而无需了解集合内部的结构。用iter和next手动操作tWECLtEtt迭代器。

    python生成器函数yield与yield from使用

     for循环无法获取到迭代的值,因为迭代器已遍历完所有的值。

    2、generator生成器里可以放什么类型的数据?

    在生成器中,可以放置各种类型的东西,包括 yield 语句、函数调用、表达式、迭代器以及 yield from 表达式。这些元素共同构成了生成器函数的功能,使其能够高效地处理大型数据集、无限序列或需要惰性计算的场景。

    3、怎么使用generator类型的数据?

    generator是一种特殊的迭代器。因为iterator可用next,不可以用for循环。但是gerater两者都可以用。

    1、获取值的方式

    def my_generator():
        yield 1
        yield 2
        yield 3
    
    # 使用 for 循环迭代生成器
    for value in my_generator():
        print(value)
    
    
    # 使用 next() 函数
    gen = my_generator()
    print(next(gen))  # 输出:1
    print(next(gen))  # 输出:2
    print(next(gen))  # 输出:3

    2、可直接用于迭代函数,如sum()max()min() 等。

    gen = my_generator()
    total = sum(gen)
    print(total)  # 输出:6

    3、generator和其他类型的互相转换

    1、generator转换成其他类型(例如列表、元祖、集合)。如下代码

    def my_generator():
        yield 1
        yield 2
        yield 3
    
    gen = my_generator()
    my_list = list(gen)  # 或者 tuple(gen)、set(gen)
    print(my_list)  # 输出:[1, 2, 3]
    

    2、其他类型转换成generator类型

    (1)使用生成器表达式

    my_list = [1, 2, 3, 4, 5]
    gen = (x for x in my_list)

    (2)函数中使用yield

    def my_generator():
        yield 1
        yield 2
        yield 3
    
    gen = my_generator()

    三、yield from使用

    1、如果解读如下代码段?

    def nested_generator():
        yield 1
        yield 2
        yield 3
    
    def main_generator():
        yield 'Start'
        yield from nested_generator()
        yield 'End'
    
    def ok_yiled_from():
        for item in main_generator():
            print(item)
    
    ok_yiled_from()
    
    
    输出:
        Start
        1
        2
        3
        End

    1、ok_yiled_from调用main_generator返回一个generator,进行for循环。

    2、调python用main_generator中的yield 'Start',则生成Start并返回给调用者,执行到该句返回,并暂停

    3、调用main_generator中的yield from nested_generator(),使用yield from调用另外一个生成器函数nested_generator()(问题:yield from后面只能跟生成器函数吗?)。

    main_generator生成器函数将php执行权转交给nested_generator(),并在其中生成的值被产生时,直接返回main_generator生成器函数的调用者ok_yiled_from。main_generator暂停执行,等待nested_generator执行完毕。

    4、当nested_generator()生成器函数执行完毕后并没有更多值可产生时,main_generator生成器函数会继续执行,yield 'End' 产出 End值。此时,生成器函数的执行完全终止。

    这样,整个过程就形成了一个流程,生成器函数逐步生成值并在必要时暂停执行,而在 yield from 语句中调用的嵌套生成器负责生成中间的值。这种机制使得生成器函数能够高效地处理大量数据或延迟计算,从而节省内存和提高性能。

    2、yield from后面要求跟什么?

    yield from后面可以跟可迭代对象或者生成器generator。

    1)跟任何可迭代对象iterable,包括列表、元组、集合、字典的键或值、字符串等

    例如:

    def list_generator():
        my_list = [1, 2, 3, 4, 5]
        yield from my_list
    
    gen = list_generator()
    for value in gen:
        print(value)android
    

    2)跟生成器generator

    如上面例子中的nested_generator()

    3、yield后面跟什么?

    yeild后面跟:不同类型单独的值、任何不同类型的表达式、语句。

    def calculate_square(x):
        return x * x
    
    def my_generator():
        yield 1
        yield "Hello"
        yield 3.14
        yield 10 * 2
        yield calculate_square(2)
    
    gen = my_generator()
    print(next(gen))  # 输出:1
    print(next(gen))  # 输出:hello
    print(next(gen))  # 输出:3.14
    print(next(gen))  # 输出:20
    print(next(gen))  # 输出:4

    4、yield from和yield的相同和不同点是什么

    相同点:

    1)都是用来返回和暂停

    2)都是生成器函数,返回generator对象。

    不同点:

    1)返回给的对象不一样。

    (也就是:用yield for和yiled修饰函数有什么不同?也就是返回给的函数不一样。)

    yield产出的值返回给当前调用的函数。yiled from返回给当前调用函数的上一个调用函数。

    • a. yield产生一个值,并将起返回给调用者;
    • b. yield from 用于委托部分工作给另外一个生成器。它在生成器中使用,可以将另外一个生成器/迭代器产出的值直接返回给调用者。

    代码说明如下:

    def generate_list():
        return [1,3,4,5]
    
    def while_yiled():
        for i in range(0,2):
            yield generate_list()
    
    def while_yiled_from():
        for i in range(0,2):
            yield from generate_list()
    
    print(list(while_yiled()))
    print(list(while_yiled_from()))
    
    输出:
    [[1, 3, 4, 5], [1, 3, 4, 5]]
    [1, 3, 4, 5, 1, 3, 4, 5]

    解读如上代码:

    1、def while_yiled(): ...

    生成器函数while_yiled() ,每次循环中生成[1,3,4,5], 把作为一个单独的元素返回给调用者print()函数,2次循环,得到了2个[1,3,4,5]元素,即结果为[[1,3,4,5]],[1,3,4,5]]]

    2、while_yiled_from()

    编程客栈

    另外一个生成器函数while_yiled_from(),每次循环生成[1,3,4,5],直接把每个元素逐个返回给调用者print(),2次循环,得到了共8个元素,即结果为[1,3,4,5,1,3,4,5]

    总结:

    while_yiled()和while_yiled_from()两个生成器函数中,使用了不同的生成器特性:

    • yiled返回整个结果给调用者。
    • yiled form直接将返回的值逐个返回给调用者。

    5、解读如下代码

    def nested_generator():
        yield 1
        yield 2
        yield 3
    
    def main_generator_yeild_form():
        yield 'Start'
        yield from nested_generator()
        yield 'End'
    
    
    def main_generator_no_yield():
        yield 'Start'
        nested_generator()
        yield 'End'
    
    def main_generator_only_yield():
        yield 'Start'
        yield nested_generator()
        yield 'End'
    
    
    def try_yiled_from():
        for item in main_generator_yeild_form():
            print(item)
        print('main_generator_yeild_form\n\n')
    
    
        for item in main_generator_only_yield():
            print(item)
        print('main_generator_only_yield\n\n')
    
    
        for item in main_generator_no_yield():
            print(item)
        print('main_generator_no_yield\n\n')
    
    
    try_yiled_from()
    
    输出:
    Start
    1
    2
    3
    End
    main_generator_yeild_form
    
    
    Start
    <generator object nested_generator at 0x7fa9f4528930>
    End
    main_generator_only_yield
    
    
    Start
    End
    main_generator_no_yield

     1、def main_generator_yeild_form():...

    生成器函数,使用yeild from 委托 nested_generator() 生成器进行工作,将返回的每个元素直接逐个给了调用者try_yiled_from中的main_generator_yeild_form(),所以结果就是:start、1、2、3、end

     2、def main_generator_no_yield():...

    生成器函数,因为调用nested_generato()是并没有使用yield、yield from,所以并没有将nested_generato()产生的值返回给调用者。nested_generato()仍然会产生值,但是在调用者try_yiled_from中的main_generator_no_yield()中,并没有返回,或者捕获到。因此它的值不会被输出。废值,没用过。

     3、def main_generator_only_yield():...

    生成器函数,调用nested_generato()是使用yield,所以返回的是生成器本身,并不是nested_generato()产生的值,所以输出结果为start, nested_generato()生成器对象,end。

    总结

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持编程客栈(www.devze.com)。

    0

    上一篇:

    下一篇:

    精彩评论

    暂无评论...
    验证码 换一张
    取 消

    最新开发

    开发排行榜