生成器

作者: 人生苦短啊 | 来源:发表于2018-09-06 13:15 被阅读0次

一. 生成器的启动方式

协程的内部是用生成器实现的, 学好了生成器会更加理解协程的原理

#1. 生成器不只可以产出值,还可以接收值
def gen_func():
    #1. 可以产出值, 2. 可以接收值(调用方传递进来的值)
    html = yield "http://projectsedu.com"
    print(html)
    yield 2
    return "bobby"

if __name__ == "__main__":
    gen = gen_func()
    # 在调用send发送非none值之前,我们必须启动一次生成器, 方式有两种1. gen.send(None), 2. next(gen)
    url = gen.send(None)   # 启动生成器
    print(url)
    html = "bobby"
    print(gen.send(html))

# http://projectsedu.com
#  bobby
#  2

二. 生成器的close方法

这么做next(gen)会向 main 抛出StopIteration异常

def gen_func():
    yield "http://projectsedu.com"
    yield 2
    yield 3
    return "bobby"

if __name__ == "__main__":
    gen = gen_func()
    print(next(gen))
    gen.close()
    next(gen)
    print("bobby")

三. 生成器的throw方法

向上抛出异常

def gen_func():
    #1. 可以产出值, 2. 可以接收值(调用方传递进来的值)
    try:
        yield "http://projectsedu.com"
    except Exception as e:
        print('我出错了')
    yield 2
    yield 3
    return "bobby"

if __name__ == "__main__":
    gen = gen_func()
    print(next(gen))
    print(gen.throw(Exception, "download error")) # 向前一个yield抛出异常
    print(next(gen))
    gen.throw(Exception, "download error")   # yield3抛出异常

# http://projectsedu.com
# 我出错了
# 2
# 3

四. yield from 方法

#python3.3新加了yield from语法
from itertools import chain  # chain 和 my_chain有一样的功能

my_list = [1,2,3]
my_dict = {
    "bobby1":"http://projectsedu.com",
    "bobby2":"http://www.imooc.com",
}

def my_chain(*args, **kwargs):
    for my_iterable in args:
        yield from my_iterable
        # for value in my_iterable:
        #     yield value

for value in my_chain(my_list, my_dict, range(5,8)):
    print(value)
# 1
# 2
# 3
# bobby1
# bobby2
# 5
# 6
# 7

五. yield from 模仿协程示例

main 调用方, middle(委托生成器) sales_sum子生成器, yield from会在调用方与子生成器之间建立一个双向通道

final_result = {}

def sales_sum(pro_name):
    total = 0
    nums = []
    while True:
        x = yield
        print(pro_name+"销量: ", x)
        if not x:
            break
        total += x
        nums.append(x)
    return total, nums

def middle(key):
    while True:
        final_result[key] = yield from sales_sum(key)
        print(key+"销量统计完成!!.")

def main():
    data_sets = {
        "bobby牌面膜": [1200, 1500, 3000],
        "bobby牌手机": [28,55,98,108 ],
        "bobby牌大衣": [280,560,778,70],
    }
    for key, data_set in data_sets.items():
        print("start key:", key)
        m = middle(key)
        m.send(None) # 预激middle协程
        for value in data_set:
            m.send(value)   # 给协程传递每一组的值
        m.send(None)
    print("final_result:", final_result)



if __name__ == '__main__':
    main()
# start key: bobby牌面膜
# bobby牌面膜销量:  1200
# bobby牌面膜销量:  1500
# bobby牌面膜销量:  3000
# bobby牌面膜销量:  None
# bobby牌面膜销量统计完成!!.
# start key: bobby牌手机
# bobby牌手机销量:  28
# bobby牌手机销量:  55
# bobby牌手机销量:  98
# bobby牌手机销量:  108
# bobby牌手机销量:  None
# bobby牌手机销量统计完成!!.
# start key: bobby牌大衣
# bobby牌大衣销量:  280
# bobby牌大衣销量:  560
# bobby牌大衣销量:  778
# bobby牌大衣销量:  70
# bobby牌大衣销量:  None
# bobby牌大衣销量统计完成!!.
# final_result: {'bobby牌面膜': (5700, [1200, 1500, 3000]), 'bobby牌手机': (289, [28, 55, 98, 108]), 'bobby牌大衣': (1688, [280, 560, 778, 70])}

相关文章

  • 15.生成器generator

    目录:1.生成器介绍2.生成器举例3.生成器应用 1.生成器介绍 生成器指的是生成器对象,可以由生成器表达式得到,...

  • 2018-07-16

    ## 1\. 生成器和生成器函数 ``` 生成器的本质就是迭代器 生成器的三种创建办法: 1.通过生成器函数 ...

  • 第014篇:三大神器之生成器

    Python的三大神器:装饰器、迭代器、生成器 1、生成器 1.1、什么是生成器 生成器就是迭代器的一种;生成器作...

  • Python 生成器函数

    一、生成器 生成器指的是生成器对象,可由生成器表达式得到,也可使用 yield 关键字得到一个生成器函数,调用这个...

  • tornado协程的工作原理

    包含yield语句的函数是一个生成器。所有的生成器都是异步的。当我们调用生成器函数的时候,生成器函数返回一个生成器...

  • python 生成器小结

    作者:邵正将 来源:PytLab 在python中生成器可以很方便的实现迭代协议。生成器通过生成器函数产生,生成器...

  • ES6 Generators

    生成器函数 生成器函数以function*标注 yield关键字,会暂停生成器的执行,在之后可以继续执行 生成器的...

  • ES6 生成器Generator

    生成器 生成器(Generators): 一个更好的方法来构建遍历器。 --- 生成器和迭代器 生成器就是一类...

  • 生成器

    生成器指的是生成器对象,可以有生成器表达式获得,也可以由yield关键字得到一个生成器,调用这个函数得到一个生成器...

  • Python 入门之 Python三大器 之 生成器

    Python 入门之 Python三大器 之 生成器 1、生成器 (1)什么是生成器? 核心:生成器的本质就是一个...

网友评论

      本文标题:生成器

      本文链接:https://www.haomeiwen.com/subject/bsbtgftx.html